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

io.hyperfoil.tools.horreum.svc.ChangesServiceImpl Maven / Gradle / Ivy

There is a newer version: 0.16.3
Show newest version
package io.hyperfoil.tools.horreum.svc;

import java.util.ArrayList;
import java.util.List;

import jakarta.annotation.PostConstruct;
import jakarta.annotation.security.PermitAll;
import jakarta.enterprise.context.ApplicationScoped;
import jakarta.inject.Inject;
import jakarta.persistence.EntityManager;
import jakarta.ws.rs.HeaderParam;
import jakarta.ws.rs.core.Response;

import org.eclipse.microprofile.config.Config;
import org.eclipse.microprofile.config.ConfigProvider;
import org.hibernate.Session;
import org.hibernate.query.NativeQuery;

import com.fasterxml.jackson.databind.JsonNode;

import io.hyperfoil.tools.horreum.api.changes.Target;
import io.hyperfoil.tools.horreum.api.internal.services.ChangesService;
import io.hyperfoil.tools.horreum.entity.alerting.ChangeDAO;
import io.hyperfoil.tools.horreum.entity.alerting.DataPointDAO;
import io.hyperfoil.tools.horreum.entity.alerting.VariableDAO;
import io.hyperfoil.tools.horreum.server.WithRoles;

/**
 * This service is a backend for the Changes report panels
 */
@PermitAll
@ApplicationScoped
public class ChangesServiceImpl implements ChangesService {
    @Inject
    EntityManager em;

    private final List allowedOrigins = new ArrayList<>();

    @PostConstruct
    void init() {
        Config config = ConfigProvider.getConfig();
        String baseUrl = config.getValue("horreum.url", String.class);
        allowedOrigins.add(getOrigin(baseUrl));
        if (baseUrl.contains("//localhost")) {
            // allow live-coding port
            int colonIndex = baseUrl.indexOf(':', 8);
            int devServerPort = config.getOptionalValue("quarkus.quinoa.dev-server.port", int.class).orElse(-1);
            if (devServerPort > 0) {
                allowedOrigins.add(baseUrl.substring(0, colonIndex < 0 ? baseUrl.length() : colonIndex) + ":" + devServerPort);
            }
        }
        config.getOptionalValue("horreum.internal.url", String.class).ifPresent(url -> allowedOrigins.add(getOrigin(url)));
    }

    private String getOrigin(String baseUrl) {
        int slashIndex = baseUrl.indexOf('/', 8); // skip http(s)://
        return baseUrl.substring(0, slashIndex < 0 ? baseUrl.length() : slashIndex);
    }

    @WithRoles
    @Override
    public String[] search(Target query) {
        return VariableDAO. listAll().stream().map(v -> String.valueOf(v.id)).toArray(String[]::new);
    }

    @WithRoles
    @Override
    public List query(Query query) {
        if (query == null) {
            throw ServiceException.badRequest("No query");
        } else if (query.range == null || query.range.from == null || query.range.to == null) {
            throw ServiceException.badRequest("Invalid time range");
        }
        List result = new ArrayList<>();
        for (Target target : query.targets) {
            if (target.type != null && !target.type.equals("timeseries")) {
                throw ServiceException.badRequest("Tables are not implemented");
            }
            String tq = target.target;
            JsonNode fingerprint = null;
            int semicolon = tq.indexOf(';');
            if (semicolon >= 0) {
                fingerprint = Util.parseFingerprint(tq.substring(semicolon + 1));
                tq = tq.substring(0, semicolon);
            }
            int variableId = parseVariableId(tq);
            if (variableId < 0) {
                throw ServiceException.badRequest("Target must be variable ID");
            }
            VariableDAO variable = VariableDAO.findById(variableId);
            String variableName = String.valueOf(variableId);
            if (variable != null) {
                variableName = variable.name;
            }
            TimeseriesTarget tt = new TimeseriesTarget();
            tt.target = variableName;
            tt.variableId = variableId;
            result.add(tt);

            StringBuilder sql = new StringBuilder("WITH dp AS (")
                    .append("(SELECT * FROM datapoint WHERE variable_id = ?1 AND timestamp BETWEEN ?2 AND ?3)");
            if (query.range.oneBeforeAndAfter) {
                sql.append(
                        "UNION (SELECT * FROM datapoint WHERE variable_id = ?1 AND timestamp < ?2 ORDER BY timestamp DESC LIMIT 1) ")
                        .append("UNION (SELECT * FROM datapoint WHERE variable_id = ?1 AND timestamp > ?3 ORDER BY timestamp LIMIT 1)");
            }
            sql.append(") SELECT dp.* FROM dp ");
            if (fingerprint != null) {
                sql.append(
                        "LEFT JOIN fingerprint fp ON fp.dataset_id = dp.dataset_id WHERE json_equals(fp.fingerprint, (?4)::jsonb) ");
            }
            sql.append("ORDER BY timestamp ASC");
            NativeQuery nativeQuery = em.unwrap(Session.class)
                    .createNativeQuery(sql.toString(), DataPointDAO.class)
                    .setParameter(1, variableId)
                    .setParameter(2, query.range.from)
                    .setParameter(3, query.range.to);
            if (fingerprint != null) {
                nativeQuery.setParameter(4, fingerprint.toString());
            }
            List datapoints = nativeQuery.getResultList();
            for (DataPointDAO dp : datapoints) {
                tt.datapoints.add(new Number[] { dp.value, dp.timestamp.toEpochMilli(), /* non-standard! */ dp.dataset.id });
            }
        }
        return result;
    }

    private int parseVariableId(String target) {
        int variableId;
        try {
            variableId = Integer.parseInt(target);
        } catch (NumberFormatException e) {
            // TODO: support test name/variable name?
            variableId = -1;
        }
        return variableId;
    }

    @Override
    public Response annotations(@HeaderParam("Origin") String origin) {
        Response.ResponseBuilder response = Response.ok()
                .header("Access-Control-Allow-Headers", "accept, content-type")
                .header("Access-Control-Allow-Methods", "POST")
                .header("Vary", "Origin");
        for (String allowed : allowedOrigins) {
            if (allowed.equals(origin)) {
                return response.header("Access-Control-Allow-Origin", allowed).build();
            }
        }
        return response.header("Access-Control-Allow-Origin", allowedOrigins.get(0)).build();
    }

    @Override
    @WithRoles
    public List annotations(AnnotationsQuery query) {
        if (query == null) {
            throw ServiceException.badRequest("No query");
        } else if (query.range == null || query.range.from == null || query.range.to == null) {
            throw ServiceException.badRequest("Invalid time range");
        }
        List annotations = new ArrayList<>();
        String tq = query.annotation.query;
        JsonNode fingerprint = null;
        int semicolon = tq.indexOf(';');
        if (semicolon >= 0) {
            fingerprint = Util.parseFingerprint(tq.substring(semicolon + 1));
            tq = tq.substring(0, semicolon);
        }
        int variableId = parseVariableId(tq);
        if (variableId < 0) {
            throw ServiceException.badRequest("Query must be variable ID");
        }
        StringBuilder sql = new StringBuilder("SELECT change.* FROM change ");
        if (fingerprint != null) {
            sql.append(" JOIN fingerprint fp ON fp.dataset_id = change.dataset_id ");
        }
        sql.append(" WHERE variable_id = ?1 AND timestamp BETWEEN ?2 AND ?3 ");
        if (fingerprint != null) {
            sql.append("AND json_equals(fp.fingerprint, (?4)::jsonb)");
        }
        NativeQuery nativeQuery = em.unwrap(Session.class).createNativeQuery(sql.toString(), ChangeDAO.class)
                .setParameter(1, variableId)
                .setParameter(2, query.range.from)
                .setParameter(3, query.range.to);
        if (fingerprint != null) {
            nativeQuery.setParameter(4, fingerprint.toString());
        }

        List changes = nativeQuery.getResultList();
        for (ChangeDAO change : changes) {
            annotations.add(createAnnotation(change));
        }
        return annotations;
    }

    private AnnotationDefinition createAnnotation(ChangeDAO change) {
        StringBuilder content = new StringBuilder("Variable: ").append(change.variable.name);
        if (change.variable.group != null) {
            content.append(" (group ").append(change.variable.group).append(")");
        }
        content.append("
").append(change.description).append("
Confirmed: ").append(change.confirmed); return new AnnotationDefinition("Change in run " + change.dataset.run.id + "/" + change.dataset.ordinal, content.toString(), false, change.timestamp.toEpochMilli(), 0, new String[0], change.id, change.variable.id, change.dataset.run.id, change.dataset.ordinal); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy