io.hyperfoil.tools.horreum.svc.ChangesServiceImpl Maven / Gradle / Ivy
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