Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
io.hyperfoil.tools.horreum.svc.AlertingServiceImpl Maven / Gradle / Ivy
package io.hyperfoil.tools.horreum.svc;
import java.time.Instant;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
import java.util.Set;
import java.util.TreeMap;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.function.BiConsumer;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.stream.Collectors;
import jakarta.annotation.security.PermitAll;
import jakarta.annotation.security.RolesAllowed;
import jakarta.enterprise.context.ApplicationScoped;
import jakarta.inject.Inject;
import jakarta.persistence.EntityManager;
import jakarta.persistence.NoResultException;
import jakarta.persistence.PersistenceException;
import jakarta.persistence.Query;
import jakarta.persistence.Tuple;
import jakarta.transaction.TransactionManager;
import jakarta.transaction.Transactional;
import jakarta.ws.rs.WebApplicationException;
import jakarta.ws.rs.core.Response;
import org.eclipse.microprofile.config.inject.ConfigProperty;
import org.hibernate.Hibernate;
import org.hibernate.Session;
import org.hibernate.query.NativeQuery;
import org.hibernate.type.StandardBasicTypes;
import org.jboss.logging.Logger;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.node.ArrayNode;
import com.fasterxml.jackson.databind.node.JsonNodeFactory;
import com.fasterxml.jackson.databind.node.ObjectNode;
import io.hyperfoil.tools.horreum.api.alerting.*;
import io.hyperfoil.tools.horreum.api.changes.Dashboard;
import io.hyperfoil.tools.horreum.api.changes.Target;
import io.hyperfoil.tools.horreum.api.data.*;
import io.hyperfoil.tools.horreum.api.data.changeDetection.ChangeDetectionModelType;
import io.hyperfoil.tools.horreum.api.internal.services.AlertingService;
import io.hyperfoil.tools.horreum.bus.AsyncEventChannels;
import io.hyperfoil.tools.horreum.bus.BlockingTaskDispatcher;
import io.hyperfoil.tools.horreum.changedetection.ChangeDetectionException;
import io.hyperfoil.tools.horreum.changedetection.ChangeDetectionModel;
import io.hyperfoil.tools.horreum.changedetection.ChangeDetectionModelResolver;
import io.hyperfoil.tools.horreum.changedetection.ModelType;
import io.hyperfoil.tools.horreum.entity.FingerprintDAO;
import io.hyperfoil.tools.horreum.entity.PersistentLogDAO;
import io.hyperfoil.tools.horreum.entity.alerting.*;
import io.hyperfoil.tools.horreum.entity.changeDetection.ChangeDetectionLogDAO;
import io.hyperfoil.tools.horreum.entity.data.DatasetDAO;
import io.hyperfoil.tools.horreum.entity.data.TestDAO;
import io.hyperfoil.tools.horreum.hibernate.IntArrayType;
import io.hyperfoil.tools.horreum.hibernate.JsonBinaryType;
import io.hyperfoil.tools.horreum.mapper.*;
import io.hyperfoil.tools.horreum.server.WithRoles;
import io.quarkus.hibernate.orm.panache.PanacheEntityBase;
import io.quarkus.panache.common.Parameters;
import io.quarkus.runtime.Startup;
import io.quarkus.scheduler.Scheduled;
import io.quarkus.security.identity.SecurityIdentity;
import io.vertx.core.Vertx;
@ApplicationScoped
@Startup
public class AlertingServiceImpl implements AlertingService {
private static final Logger log = Logger.getLogger(AlertingServiceImpl.class);
//@formatter:off
private static final String LOOKUP_TIMESTAMP =
"""
SELECT timeline_function,
(CASE
WHEN jsonb_array_length(timeline_labels) = 1 THEN jsonb_agg(lv.value)->0
ELSE COALESCE(jsonb_object_agg(label.name, lv.value) FILTER (WHERE label.name IS NOT NULL), '{}'::jsonb)
END) as value
FROM test
JOIN label ON json_contains(timeline_labels, label.name)
JOIN label_values lv ON label.id = lv.label_id
WHERE test.id = ?1 AND lv.dataset_id = ?2
AND timeline_labels IS NOT NULL
AND jsonb_typeof(timeline_labels) = 'array'
AND jsonb_array_length(timeline_labels) > 0
GROUP BY timeline_function, timeline_labels
""";
private static final String LOOKUP_VARIABLES =
"""
SELECT
var.id as variableId,
var.name,
var.\"group\",
var.calculation,
jsonb_array_length(var.labels) AS numLabels,
(CASE
WHEN jsonb_array_length(var.labels) = 1 THEN jsonb_agg(lv.value)->0
ELSE COALESCE(jsonb_object_agg(label.name, lv.value) FILTER (WHERE label.name IS NOT NULL), '{}'::jsonb)
END) AS value
FROM variable var
LEFT JOIN label ON json_contains(var.labels, label.name)
LEFT JOIN label_values lv ON label.id = lv.label_id
WHERE var.testid = ?1
AND lv.dataset_id = ?2
GROUP BY var.id, var.name, var.\"group\", var.calculation
""";
private static final String LOOKUP_RULE_LABEL_VALUES =
"""
SELECT
mdr.id AS rule_id,
mdr.condition,
(CASE
WHEN mdr.labels IS NULL OR jsonb_array_length(mdr.labels) = 0 THEN NULL
WHEN jsonb_array_length(mdr.labels) = 1 THEN jsonb_agg(lv.value)->0
ELSE COALESCE(jsonb_object_agg(label.name, lv.value) FILTER (WHERE label.name IS NOT NULL), '{}'::jsonb)
END) as value
FROM missingdata_rule mdr
LEFT JOIN label ON json_contains(mdr.labels, label.name)
LEFT JOIN label_values lv ON label.id = lv.label_id AND lv.dataset_id = ?1
WHERE mdr.test_id = ?2
GROUP BY rule_id, mdr.condition
""";
private static final String LOOKUP_LABEL_VALUE_FOR_RULE =
"""
SELECT
(CASE
WHEN mdr.labels IS NULL OR jsonb_array_length(mdr.labels) = 0 THEN NULL
WHEN jsonb_array_length(mdr.labels) = 1 THEN jsonb_agg(lv.value)->0
ELSE COALESCE(jsonb_object_agg(label.name, lv.value) FILTER (WHERE label.name IS NOT NULL), '{}'::jsonb)
END) as value
FROM missingdata_rule mdr
LEFT JOIN label ON json_contains(mdr.labels, label.name)
LEFT JOIN label_values lv ON label.id = lv.label_id AND lv.dataset_id = ?1
WHERE mdr.id = ?2
GROUP BY mdr.labels
""";
private static final String LOOKUP_RECENT =
"""
SELECT
DISTINCT ON(mdr.id) mdr.id,
mdr.test_id,
mdr.name,
mdr.maxstaleness,
rr.timestamp
FROM missingdata_rule mdr
LEFT JOIN missingdata_ruleresult rr ON mdr.id = rr.rule_id
WHERE last_notification IS NULL
OR EXTRACT(EPOCH FROM last_notification) * 1000 < EXTRACT(EPOCH FROM current_timestamp) * 1000 - mdr.maxstaleness
ORDER BY mdr.id, timestamp DESC
""";
private static final String FIND_LAST_DATAPOINTS =
"""
SELECT
DISTINCT ON(variable_id) variable_id AS variable,
EXTRACT(EPOCH FROM timestamp) * 1000 AS timestamp
FROM datapoint dp
LEFT JOIN fingerprint fp ON fp.dataset_id = dp.dataset_id
WHERE
((fp.fingerprint IS NULL AND (?1)::jsonb IS NULL) OR json_equals(fp.fingerprint, (?1)::jsonb))
AND variable_id = ANY(?2)
ORDER BY variable_id, timestamp DESC
""";
//@formatter:on
private static final Instant LONG_TIME_AGO = Instant.ofEpochSecond(0);
private static final Instant VERY_DISTANT_FUTURE = Instant.parse("2666-06-06T06:06:06.00Z");
@Inject
TestServiceImpl testService;
@Inject
EntityManager em;
@Inject
BlockingTaskDispatcher messageBus;
@Inject
SecurityIdentity identity;
@ConfigProperty(name = "horreum.alerting.updateLabel.retries", defaultValue = "5")
Integer labelCalcRetries;
@Inject
TransactionManager tm;
@Inject
Vertx vertx;
@Inject
NotificationServiceImpl notificationService;
@Inject
TimeService timeService;
@Inject
ServiceMediator mediator;
@Inject
Session session;
@Inject
ChangeDetectionModelResolver modelResolver;
static ConcurrentHashMap retryCounterSet = new ConcurrentHashMap<>();
// entries can be removed from timer thread while normally this is updated from one of blocking threads
private final ConcurrentMap recalcProgress = new ConcurrentHashMap<>();
// A new datapoint invalidates anything past its timestamp. Any attempt to recalculate starts
// at the timestamp.
private final ConcurrentMap validUpTo = new ConcurrentHashMap<>();
static {
System.setProperty("polyglot.engine.WarnInterpreterOnly", "false");
}
@WithRoles(extras = Roles.HORREUM_SYSTEM)
@Transactional
public void onLabelsUpdated(Dataset.LabelsUpdatedEvent event) {
boolean sendNotifications;
DataPointDAO.delete("dataset.id", event.datasetId);
DatasetDAO dataset = DatasetDAO.findById(event.datasetId);
if (dataset == null) {
// The run is not committed yet?
// Retry `horreum.alerting.updateLabel.retries` times before logging a warning
retryCounterSet.putIfAbsent(event.datasetId, new AtomicInteger(0));
int retryCounter = retryCounterSet.get(event.datasetId).getAndIncrement();
if (retryCounter < labelCalcRetries) {
log.infof("Retrying labels update for dataset %d, attempt %d/%d", event.datasetId, retryCounter,
this.labelCalcRetries);
vertx.setTimer(1000, timerId -> messageBus.executeForTest(event.datasetId, () -> Util.withTx(tm, () -> {
onLabelsUpdated(event);
return null;
})));
return;
} else {
//we have retried `horreum.alerting.updateLabel.retries` number of times, log a warning and stop retrying
log.warnf("Unsuccessfully retried updating labels %d times for dataset %d. Stopping", this.labelCalcRetries,
event.datasetId);
retryCounterSet.remove(event.datasetId);
return;
}
}
if (event.isRecalculation) {
sendNotifications = false;
} else {
try {
sendNotifications = (Boolean) em.createNativeQuery("SELECT notificationsenabled FROM test WHERE id = ?")
.setParameter(1, dataset.testid).getSingleResult();
} catch (NoResultException e) {
sendNotifications = true;
}
}
Recalculation recalculation = new Recalculation();
recalculation.clearDatapoints = true;
recalculation.lastDatapoint = true;
recalculateDatapointsForDataset(dataset, sendNotifications, false, recalculation);
recalculateMissingDataRules(dataset);
}
private void recalculateMissingDataRules(DatasetDAO dataset) {
MissingDataRuleResultDAO.deleteForDataset(dataset.id);
List ruleValues = session
.createNativeQuery(LOOKUP_RULE_LABEL_VALUES, Object[].class)
.setParameter(1, dataset.id).setParameter(2, dataset.testid)
.addScalar("rule_id", StandardBasicTypes.INTEGER)
.addScalar("condition", StandardBasicTypes.TEXT)
.addScalar("value", JsonBinaryType.INSTANCE)
.getResultList();
Util.evaluateWithCombinationFunction(ruleValues, row -> (String) row[1], row -> (JsonNode) row[2],
(row, result) -> {
int ruleId = (int) row[0];
if (result.isBoolean()) {
if (result.asBoolean()) {
createMissingDataRuleResult(dataset, ruleId);
}
} else {
logMissingDataMessage(dataset, PersistentLogDAO.ERROR,
"Result for missing data rule %d, dataset %d is not a boolean: %s", ruleId, dataset.id, result);
}
},
// Absence of condition means that this dataset is taken into account. This happens e.g. when value == NULL
row -> createMissingDataRuleResult(dataset, (int) row[0]),
(row, exception, code) -> logMissingDataMessage(dataset, PersistentLogDAO.ERROR,
"Exception evaluating missing data rule %d, dataset %d: '%s' Code: %s ", row[0], dataset.id,
exception.getMessage(), code),
output -> logMissingDataMessage(dataset, PersistentLogDAO.DEBUG,
"Output while evaluating missing data rules for dataset %d: '%s'", dataset.id, output));
}
private void createMissingDataRuleResult(DatasetDAO dataset, int ruleId) {
new MissingDataRuleResultDAO(ruleId, dataset.id, dataset.start).persist();
}
private void recalculateDatapointsForDataset(DatasetDAO dataset, boolean notify, boolean debug,
Recalculation recalculation) {
log.debugf("Analyzing dataset %d (%d/%d)", (long) dataset.id, (long) dataset.run.id, dataset.ordinal);
TestDAO test = TestDAO.findById(dataset.testid);
if (test == null) {
log.errorf("Cannot load test ID %d", dataset.testid);
return;
}
if (!testFingerprint(dataset, test.fingerprintFilter)) {
return;
}
emitDatapoints(dataset, notify, debug, recalculation);
}
private boolean testFingerprint(DatasetDAO dataset, String filter) {
if (filter == null || filter.isBlank()) {
return true;
}
Optional result = session
.createNativeQuery("SELECT fp.fingerprint FROM fingerprint fp WHERE dataset_id = ?1")
.setParameter(1, dataset.id)
.addScalar("fingerprint", JsonBinaryType.INSTANCE)
.getResultStream().findFirst();
JsonNode fingerprint;
if (result.isPresent()) {
fingerprint = result.get();
if (fingerprint.isObject() && fingerprint.size() == 1) {
fingerprint = fingerprint.elements().next();
}
} else {
fingerprint = JsonNodeFactory.instance.nullNode();
}
boolean testResult = Util.evaluateTest(filter, fingerprint,
value -> {
logCalculationMessage(dataset, PersistentLogDAO.ERROR,
"Evaluation of fingerprint failed: '%s' is not a boolean", value);
return false;
},
(code, e) -> logCalculationMessage(dataset, PersistentLogDAO.ERROR,
"Evaluation of fingerprint filter failed: '%s' Code:%s ", e.getMessage(), code),
output -> logCalculationMessage(dataset, PersistentLogDAO.DEBUG,
"Output while evaluating fingerprint filter: %s ", output));
if (!testResult) {
logCalculationMessage(dataset, PersistentLogDAO.DEBUG, "Fingerprint %s was filtered out.", fingerprint);
}
return testResult;
}
void exportTest(TestExport test) {
test.variables = VariableDAO. list("testId", test.id).stream().map(VariableMapper::from)
.collect(Collectors.toList());
test.missingDataRules = MissingDataRuleDAO. list("test.id", test.id).stream()
.map(MissingDataRuleMapper::from).collect(Collectors.toList());
}
void importVariables(TestExport test) {
for (var v : test.variables) {
VariableDAO variable = VariableMapper.to(v);
variable.ensureLinked();
if (variable.id != null && VariableDAO.findById(variable.id) == null) {
int prevId = variable.id;
variable.flushIds();
variable.persist();
test.updateExperimentsVariableId(prevId, variable.id);
} else
em.merge(variable);
}
}
void importMissingDataRules(TestExport test) {
for (var rule : test.missingDataRules) {
if (MissingDataRuleDAO.findById(rule.id) != null) {
em.merge(MissingDataRuleMapper.to(rule));
} else {
rule.id = null;
em.persist(MissingDataRuleMapper.to(rule));
}
}
}
public static class VariableData {
public int variableId;
public String name;
public String group;
public String calculation;
public int numLabels;
public JsonNode value;
public String fullName() {
return (group == null || group.isEmpty()) ? name : group + "/" + name;
}
}
private void emitDatapoints(DatasetDAO dataset, boolean notify, boolean debug, Recalculation recalculation) {
Set missingValueVariables = new HashSet<>();
List values = session.createNativeQuery(LOOKUP_VARIABLES, Tuple.class)
.setParameter(1, dataset.testid)
.setParameter(2, dataset.id)
.addScalar("variableId", StandardBasicTypes.INTEGER)
.addScalar("name", StandardBasicTypes.TEXT)
.addScalar("group", StandardBasicTypes.TEXT)
.addScalar("calculation", StandardBasicTypes.TEXT)
.addScalar("numLabels", StandardBasicTypes.INTEGER)
.addScalar("value", JsonBinaryType.INSTANCE)
.setTupleTransformer((tuples, aliases) -> {
VariableData data = new VariableData();
data.variableId = (int) tuples[0];
data.name = (String) tuples[1];
data.group = (String) tuples[2];
data.calculation = (String) tuples[3];
data.numLabels = (int) tuples[4];
data.value = (JsonNode) tuples[5];
return data;
})
.getResultList();
if (debug) {
for (VariableData data : values) {
logCalculationMessage(dataset, PersistentLogDAO.DEBUG, "Fetched value for variable %s: %s ",
data.fullName(), data.value);
}
}
List timestampList = session
.createNativeQuery(LOOKUP_TIMESTAMP, Object[].class)
.setParameter(1, dataset.testid)
.setParameter(2, dataset.id)
.addScalar("timeline_function", StandardBasicTypes.TEXT)
.addScalar("value", JsonBinaryType.INSTANCE)
.getResultList();
Instant timestamp = dataset.start;
if (!timestampList.isEmpty()) {
String timestampFunction = (String) timestampList.get(0)[0];
JsonNode value = (JsonNode) timestampList.get(0)[1];
if (timestampFunction != null && !timestampFunction.isBlank()) {
value = Util.evaluateOnce(timestampFunction, value, Util::convertToJson,
(code, throwable) -> logCalculationMessage(dataset, PersistentLogDAO.ERROR,
"Evaluation of timestamp failed: '%s' Code: %s
", throwable.getMessage(),
code),
output -> logCalculationMessage(dataset, PersistentLogDAO.DEBUG,
"Output while calculating timestamp: %s ", output));
}
timestamp = Util.toInstant(value);
if (timestamp == null) {
logCalculationMessage(dataset, PersistentLogDAO.ERROR,
"Cannot parse timestamp, must be number or ISO-8601 timestamp: %s", value);
timestamp = dataset.start;
}
}
Instant finalTimestamp = timestamp;
Util.evaluateWithCombinationFunction(values, data -> data.calculation, data -> data.value,
(data, result) -> {
Double value = Util.toDoubleOrNull(result,
error -> logCalculationMessage(dataset, PersistentLogDAO.ERROR,
"Evaluation of variable %s failed: %s", data.fullName(), error),
info -> logCalculationMessage(dataset, PersistentLogDAO.INFO, "Evaluation of variable %s: %s",
data.fullName(), info));
if (value != null) {
createDataPoint(dataset, finalTimestamp, data.variableId, value, notify, recalculation);
} else {
if (recalculation != null) {
recalculation.datasetsWithoutValue.put(dataset.id, dataset.getInfo());
}
missingValueVariables.add(data.fullName());
}
},
data -> {
if (data.numLabels > 1) {
logCalculationMessage(dataset, PersistentLogDAO.WARN,
"Variable %s has more than one label (%s) but no calculation function.", data.fullName(),
data.value.fieldNames());
}
if (data.value == null || data.value.isNull()) {
logCalculationMessage(dataset, PersistentLogDAO.INFO,
"Null value for variable %s - datapoint is not created", data.fullName());
if (recalculation != null) {
recalculation.datasetsWithoutValue.put(dataset.id, dataset.getInfo());
}
missingValueVariables.add(data.fullName());
return;
}
Double value = null;
if (data.value.isNumber()) {
value = data.value.asDouble();
} else if (data.value.isTextual()) {
try {
value = Double.parseDouble(data.value.asText());
} catch (NumberFormatException e) {
// ignore
}
}
if (value == null) {
logCalculationMessage(dataset, PersistentLogDAO.ERROR,
"Cannot turn %s into a floating-point value for variable %s", data.value, data.fullName());
if (recalculation != null) {
recalculation.errors++;
}
missingValueVariables.add(data.fullName());
} else {
createDataPoint(dataset, finalTimestamp, data.variableId, value, notify, recalculation);
}
},
(data, exception, code) -> logCalculationMessage(dataset, PersistentLogDAO.ERROR,
"Evaluation of variable %s failed: '%s' Code:%s ", data.fullName(), exception.getMessage(),
code),
output -> logCalculationMessage(dataset, PersistentLogDAO.DEBUG,
"Output while calculating variable: %s ", output));
if (!missingValueVariables.isEmpty()) {
MissingValuesEvent event = new MissingValuesEvent(dataset.getInfo(), missingValueVariables, notify);
if (mediator.testMode())
mediator.publishEvent(AsyncEventChannels.DATASET_MISSING_VALUES, dataset.testid, event);
mediator.missingValuesDataset(event);
}
DataPoint.DatasetProcessedEvent event = new DataPoint.DatasetProcessedEvent(DatasetMapper.fromInfo(dataset.getInfo()),
notify);
if (mediator.testMode())
Util.registerTxSynchronization(tm,
txStatus -> mediator.publishEvent(AsyncEventChannels.DATAPOINT_PROCESSED, dataset.testid, event));
mediator.dataPointsProcessed(event);
}
@Transactional
void createDataPoint(DatasetDAO dataset, Instant timestamp, int variableId, double value, boolean notify,
Recalculation recalculation) {
DataPointDAO dataPoint = null;
VariableDAO variableDAO = VariableDAO.findById(variableId);
if (recalculation.clearDatapoints) {
dataPoint = new DataPointDAO();
dataPoint.variable = variableDAO;
dataPoint.dataset = dataset;
dataPoint.timestamp = timestamp;
dataPoint.value = value;
dataPoint.persistAndFlush();
} else {
dataPoint = DataPointDAO.find("dataset = :dataset and variable = :variable",
Parameters.with("dataset", dataset).and("variable", variableDAO)).firstResult();
}
if (dataPoint != null) {
DataPoint.Event event = new DataPoint.Event(DataPointMapper.from(dataPoint), dataset.testid, notify);
onNewDataPoint(event, recalculation.lastDatapoint); //Test failure if we do not start a new thread and new tx
if (mediator.testMode())
Util.registerTxSynchronization(tm,
txStatus -> mediator.publishEvent(AsyncEventChannels.DATAPOINT_NEW, dataset.testid, event));
} else {
log.debugf("DataPoint for dataset %d, variable %d, timestamp %s, value %f not found", dataset.id, variableId,
timestamp, value);
}
}
private void logCalculationMessage(DatasetDAO dataSet, int level, String format, Object... args) {
logCalculationMessage(dataSet.testid, dataSet.id, level, format, args);
}
private void logCalculationMessage(int testId, int datasetId, int level, String format, Object... args) {
String msg = args.length == 0 ? format : String.format(format, args);
log.tracef("Logging %s for test %d, dataset %d: %s", PersistentLogDAO.logLevel(level), testId, datasetId, msg);
new DatasetLogDAO(em.getReference(TestDAO.class, testId), em.getReference(DatasetDAO.class, datasetId),
level, "variables", msg).persist();
}
private void logMissingDataMessage(DatasetDAO dataSet, int level, String format, Object... args) {
logMissingDataMessage(dataSet.testid, dataSet.id, level, format, args);
}
private void logMissingDataMessage(int testId, int datasetId, int level, String format, Object... args) {
String msg = args.length == 0 ? format : String.format(format, args);
log.tracef("Logging %s for test %d, dataset %d: %s", PersistentLogDAO.logLevel(level), testId, datasetId, msg);
new DatasetLogDAO(em.getReference(TestDAO.class, testId), em.getReference(DatasetDAO.class, datasetId),
level, "missingdata", msg).persist();
}
private void logChangeDetectionMessage(int testId, int datasetId, int level, String format, Object... args) {
String msg = args.length == 0 ? format : String.format(format, args);
log.tracef("Logging %s for test %d, dataset %d: %s", PersistentLogDAO.logLevel(level), testId, datasetId, msg);
new DatasetLogDAO(em.getReference(TestDAO.class, testId), em.getReference(DatasetDAO.class, datasetId),
level, "changes", msg).persist();
}
@WithRoles(extras = Roles.HORREUM_SYSTEM)
@Transactional
void onNewDataPoint(DataPoint.Event event, boolean lastDatapoint) {
DataPoint dataPoint = event.dataPoint;
if (dataPoint.variable != null && dataPoint.variable.id != null) {
VariableDAO variable = VariableDAO.findById(dataPoint.variable.id);
if (variable != null) {
log.debugf("Processing new datapoint for dataset %d at %s, variable %d (%s), value %f",
dataPoint.datasetId, dataPoint.timestamp,
variable.id, variable.name, dataPoint.value);
JsonNode fingerprint = FingerprintDAO. findByIdOptional(dataPoint.datasetId)
.map(fp -> fp.fingerprint).orElse(null);
VarAndFingerprint key = new VarAndFingerprint(variable.id, fingerprint);
log.debugf("Invalidating variable %d FP %s timestamp %s, current value is %s", variable.id, fingerprint,
dataPoint.timestamp, validUpTo.get(key));
validUpTo.compute(key, (ignored, current) -> {
if (current == null || !dataPoint.timestamp.isAfter(current.timestamp)) {
return new UpTo(dataPoint.timestamp, false);
} else {
return current;
}
});
runChangeDetection(VariableDAO.findById(variable.id), fingerprint, event.notify, true, lastDatapoint);
} else {
log.warnf("Could not process new datapoint for dataset %d at %s, could not find variable by id %d ",
dataPoint.datasetId, dataPoint.timestamp, dataPoint.variable == null ? -1 : dataPoint.variable.id);
}
} else {
log.warnf("Could not process new datapoint for dataset %d when the supplied variable or id reference is null ",
dataPoint.datasetId);
}
}
@WithRoles(extras = Roles.HORREUM_SYSTEM)
@Transactional
void tryRunChangeDetection(VariableDAO variable, JsonNode fingerprint, boolean notify, boolean lastDatapoint) {
runChangeDetection(variable, fingerprint, notify, false, lastDatapoint);
}
private void runChangeDetection(VariableDAO variable, JsonNode fingerprint, boolean notify, boolean expectExists,
boolean lastDatapoint) {
UpTo valid = validUpTo.get(new VarAndFingerprint(variable.id, fingerprint));
Instant nextTimestamp = session.createNativeQuery(
"SELECT MIN(timestamp) FROM datapoint dp LEFT JOIN fingerprint fp ON dp.dataset_id = fp.dataset_id " +
"WHERE dp.variable_id = ?1 AND (timestamp > ?2 OR (timestamp = ?2 AND ?3)) AND json_equals(fp.fingerprint, ?4)",
Instant.class)
.setParameter(1, variable.id)
.setParameter(2, valid != null ? valid.timestamp : LONG_TIME_AGO, StandardBasicTypes.INSTANT)
.setParameter(3, valid == null || !valid.inclusive)
.setParameter(4, fingerprint, JsonBinaryType.INSTANCE)
.getResultStream().filter(Objects::nonNull).findFirst().orElse(null);
if (nextTimestamp == null) {
log.debugf("No further datapoints for change detection");
return;
}
// this should happen only after reboot, let's start with last change
if (valid != null) {
int numDeleted = session.createNativeQuery("DELETE FROM change cc WHERE cc.id IN (" +
"SELECT id FROM change c LEFT JOIN fingerprint fp ON c.dataset_id = fp.dataset_id " +
"WHERE NOT c.confirmed AND c.variable_id = ?1 AND (c.timestamp > ?2 OR (c.timestamp = ?2 AND ?3)) " +
"AND json_equals(fp.fingerprint, ?4))", int.class)
.setParameter(1, variable.id)
.setParameter(2, valid.timestamp, StandardBasicTypes.INSTANT)
.setParameter(3, !valid.inclusive)
.setParameter(4, fingerprint, JsonBinaryType.INSTANCE)
.executeUpdate();
log.debugf("Deleted %d changes %s %s for variable %d, fingerprint %s", numDeleted, valid.inclusive ? ">" : ">=",
valid.timestamp, variable.id, fingerprint);
}
var changeQuery = session
.createQuery("SELECT c FROM Change c LEFT JOIN Fingerprint fp ON c.dataset.id = fp.dataset.id " +
"WHERE c.variable = ?1 AND (c.timestamp < ?2 OR (c.timestamp = ?2 AND ?3 = TRUE)) AND " +
"TRUE = function('json_equals', fp.fingerprint, ?4) " +
"ORDER by c.timestamp DESC", ChangeDAO.class);
changeQuery
.setParameter(1, variable)
.setParameter(2, valid != null ? valid.timestamp : VERY_DISTANT_FUTURE)
.setParameter(3, valid == null || valid.inclusive)
.setParameter(4, fingerprint, JsonBinaryType.INSTANCE);
ChangeDAO lastChange = changeQuery.setMaxResults(1).getResultStream().findFirst().orElse(null);
Instant changeTimestamp = LONG_TIME_AGO;
if (lastChange != null) {
log.debugf("Filtering DP between %s (change %d) and %s", lastChange.timestamp, lastChange.id, nextTimestamp);
changeTimestamp = lastChange.timestamp;
}
List dataPoints = session.createQuery(
"SELECT dp FROM DataPoint dp LEFT JOIN Fingerprint fp ON dp.dataset.id = fp.dataset.id " +
"JOIN dp.dataset " + // ignore datapoints (that were not deleted yet) from deleted datasets
"WHERE dp.variable = ?1 AND dp.timestamp BETWEEN ?2 AND ?3 " +
"AND TRUE = function('json_equals', fp.fingerprint, ?4) " +
"ORDER BY dp.timestamp DESC, dp.dataset.id DESC",
DataPointDAO.class)
.setParameter(1, variable)
.setParameter(2, changeTimestamp)
.setParameter(3, nextTimestamp)
.setParameter(4, fingerprint, JsonBinaryType.INSTANCE)
.getResultList();
// Last datapoint is already in the list
if (dataPoints.isEmpty()) {
if (expectExists) {
log.warn("The published datapoint should be already in the list");
}
} else {
int datasetId = dataPoints.get(0).getDatasetId();
for (ChangeDetectionDAO detection : ChangeDetectionDAO. find("variable", variable).list()) {
ChangeDetectionModel model = modelResolver.getModel(ChangeDetectionModelType.fromString(detection.model));
if (model == null) {
logChangeDetectionMessage(variable.testId, datasetId, PersistentLogDAO.ERROR,
"Cannot find change detection model %s", detection.model);
continue;
}
//Only run bulk models on the last datapoint, otherwise run on every datapoint
if (model.getType() == ModelType.CONTINOUS || (model.getType() == ModelType.BULK && lastDatapoint)) {
try {
model.analyze(dataPoints, detection.config, change -> {
logChangeDetectionMessage(variable.testId, datasetId, PersistentLogDAO.DEBUG,
"Change %s detected using datapoints %s", change, reversedAndLimited(dataPoints));
DatasetDAO.Info info = session
.createNativeQuery(
"SELECT id, runid as \"runId\", ordinal, testid as \"testId\" FROM dataset WHERE id = ?1",
Tuple.class)
.setParameter(1, change.dataset.id)
.setTupleTransformer((tuples, aliases) -> {
DatasetDAO.Info i = new DatasetDAO.Info();
i.id = (int) tuples[0];
i.runId = (int) tuples[1];
i.ordinal = (int) tuples[2];
i.testId = (int) tuples[3];
return i;
}).getSingleResult();
em.persist(change);
Hibernate.initialize(change.dataset.run.id);
String testName = TestDAO. findByIdOptional(variable.testId).map(test -> test.name)
.orElse("");
Change.Event event = new Change.Event(ChangeMapper.from(change), testName,
DatasetMapper.fromInfo(info), notify);
if (mediator.testMode())
Util.registerTxSynchronization(tm, txStatus -> mediator
.publishEvent(AsyncEventChannels.CHANGE_NEW, change.dataset.testid, event));
mediator.executeBlocking(() -> mediator.newChange(event));
});
} catch (ChangeDetectionException e) {
new ChangeDetectionLogDAO(variable, fingerprint, PersistentLogDAO.ERROR, e.getLocalizedMessage())
.persist();
log.error("An error occurred while running change detection!", e);
}
}
}
}
Util.doAfterCommit(tm, () -> {
validateUpTo(variable, fingerprint, nextTimestamp);
//assume not last datapoint if we have found more
messageBus.executeForTest(variable.testId, () -> tryRunChangeDetection(variable, fingerprint, notify, false));
});
}
private void validateUpTo(VariableDAO variable, JsonNode fingerprint, Instant timestamp) {
validUpTo.compute(new VarAndFingerprint(variable.id, fingerprint), (ignored, current) -> {
log.debugf("Attempt %s, valid up to %s, ", timestamp, current);
if (current == null || !current.timestamp.isAfter(timestamp)) {
return new UpTo(timestamp, true);
} else {
return current;
}
});
}
private String reversedAndLimited(List list) {
int maxIndex = Math.min(list.size() - 1, 20);
StringBuilder sb = new StringBuilder("[");
if (maxIndex < list.size() - 1) {
sb.append("..., ");
}
for (int i = maxIndex; i >= 0; --i) {
sb.append(list.get(i));
if (i != 0)
sb.append(", ");
}
return sb.append("]").toString();
}
@Override
@WithRoles
@PermitAll
public List variables(Integer testId) {
List variables;
if (testId != null) {
variables = VariableDAO.list("testId", testId);
} else {
variables = VariableDAO.listAll();
}
return variables.stream().map(VariableMapper::from).collect(Collectors.toList());
}
@Override
@WithRoles
@RolesAllowed("tester")
@Transactional
public void updateVariables(int testId, List variablesDTO) {
for (Variable v : variablesDTO) {
if (v.name == null || v.name.isBlank()) {
throw ServiceException.badRequest("Variable name is mandatory!");
}
}
try {
List variables = variablesDTO.stream().map(VariableMapper::to).collect(Collectors.toList());
List currentVariables = VariableDAO.list("testId", testId);
updateCollection(currentVariables, variables, v -> v.id, item -> {
if (item.id != null && item.id < 0) {
item.id = null;
}
if (item.changeDetection != null) {
ensureDefaults(item.changeDetection);
item.changeDetection.forEach(rd -> rd.variable = item);
item.changeDetection.stream().filter(rd -> rd.id != null && rd.id == -1).forEach(rd -> rd.id = null);
}
item.testId = testId;
item.persist(); // insert
}, (current, matching) -> {
current.name = matching.name;
current.group = matching.group;
current.labels = matching.labels;
current.calculation = matching.calculation;
if (matching.changeDetection != null) {
ensureDefaults(matching.changeDetection);
}
updateCollection(current.changeDetection, matching.changeDetection, rd -> rd.id, item -> {
if (item.id != null && item.id < 0) {
item.id = null;
}
item.variable = current;
item.persist();
current.changeDetection.add(item);
}, (crd, mrd) -> {
crd.model = mrd.model;
crd.config = mrd.config;
}, PanacheEntityBase::delete);
current.persist();
}, current -> {
DataPointDAO.delete("variable.id", current.id);
ChangeDAO.delete("variable.id", current.id);
current.delete();
});
em.flush();
} catch (PersistenceException e) {
log.error("Failed to update variables", e);
throw new WebApplicationException(e, Response.serverError().build());
}
log.debug("Variables updated, everything is fine, returning");
}
private void ensureDefaults(Set rds) {
rds.forEach(rd -> {
ChangeDetectionModel model = modelResolver.getModel(ChangeDetectionModelType.fromString(rd.model));
if (model == null) {
throw ServiceException.badRequest("Unknown model " + rd.model);
}
if (rd.config == null || rd.config.isNull() || rd.config.isMissingNode()) {
rd.config = JsonNodeFactory.instance.objectNode();
}
if (!(rd.config instanceof ObjectNode)) {
throw ServiceException.badRequest("Invalid config for model " + rd.model + " - not an object: " + rd.config);
}
for (var entry : model.config().defaults.entrySet()) {
JsonNode property = rd.config.get(entry.getKey());
if (property == null || property.isNull()) {
((ObjectNode) rd.config).set(entry.getKey(), entry.getValue());
}
}
});
}
private void updateCollection(Collection currentList, Collection newList, Function idSelector,
Consumer create, BiConsumer update, Consumer delete) {
for (Iterator iterator = currentList.iterator(); iterator.hasNext();) {
T current = iterator.next();
T matching = newList.stream().filter(v -> idSelector.apply(current).equals(idSelector.apply(v))).findFirst()
.orElse(null);
if (matching == null) {
delete.accept(current);
iterator.remove();
} else {
update.accept(current, matching);
}
}
for (T item : newList) {
if (currentList.stream().noneMatch(v -> idSelector.apply(v).equals(idSelector.apply(item)))) {
create.accept(item);
}
}
}
private DashboardInfo createChangesDashboard(int testId, String fingerprint, List variables) {
DashboardInfo info = new DashboardInfo();
info.testId = testId;
Dashboard dashboard = new Dashboard();
dashboard.title = TestDAO. findByIdOptional(testId).map(t -> t.name).orElse("Test " + testId)
+ (fingerprint.isEmpty() ? "" : ", " + fingerprint);
dashboard.tags.add(testId + ";" + fingerprint);
dashboard.tags.add("testId=" + testId);
int i = 0;
Map> byGroup = groupedVariables(variables);
for (VariableDAO variable : variables) {
dashboard.annotations.list.add(new Dashboard.Annotation(variable.name, variable.id + ";" + fingerprint));
}
for (Map.Entry> entry : byGroup.entrySet()) {
entry.getValue().sort(Comparator.comparing(v -> v.name));
Dashboard.Panel panel = new Dashboard.Panel(entry.getKey(),
new Dashboard.GridPos(12 * (i % 2), 9 * (i / 2), 12, 9));
info.panels.add(new PanelInfo(entry.getKey(),
entry.getValue().stream().map(VariableMapper::from).collect(Collectors.toList())));
for (VariableDAO variable : entry.getValue()) {
panel.targets.add(new Target(variable.id + ";" + fingerprint, "timeseries", "T" + i));
}
dashboard.panels.add(panel);
++i;
}
return info;
}
private Map> groupedVariables(List variables) {
Map> byGroup = new TreeMap<>();
for (VariableDAO variable : variables) {
byGroup.computeIfAbsent(variable.group == null || variable.group.isEmpty() ? variable.name : variable.group,
g -> new ArrayList<>()).add(variable);
}
return byGroup;
}
@Override
@WithRoles
@PermitAll
@Transactional
public DashboardInfo dashboard(int testId, String fingerprint) {
if (fingerprint == null) {
fingerprint = "";
}
List variables = VariableDAO.list("testId", testId);
return createChangesDashboard(testId, fingerprint, variables);
}
@Override
@WithRoles
@PermitAll
public List changes(int varId, String fingerprint) {
VariableDAO v = VariableDAO.findById(varId);
if (v == null) {
throw ServiceException.notFound("Variable " + varId + " not found");
}
JsonNode fp = Util.parseFingerprint(fingerprint);
if (fp == null) {
List changes = ChangeDAO.list("variable", v);
return changes.stream().map(ChangeMapper::from).collect(Collectors.toList());
}
List changes = session.createNativeQuery("""
SELECT change.*
FROM change
JOIN fingerprint fp ON change.dataset_id = fp.dataset_id
WHERE variable_id = ?1
AND json_equals(fp.fingerprint, ?2)
""", ChangeDAO.class)
.setParameter(1, varId)
.setParameter(2, fp, JsonBinaryType.INSTANCE)
.getResultList();
return changes.stream().map(ChangeMapper::from).collect(Collectors.toList());
}
@Override
@WithRoles
@RolesAllowed(Roles.TESTER)
@Transactional
public void updateChange(int id, Change apiChange) {
try {
if (id != apiChange.id) {
throw ServiceException.badRequest("Path ID and entity don't match");
}
ChangeDAO jpaChange = em.find(ChangeDAO.class, id);
if (jpaChange != null) {
jpaChange.confirmed = apiChange.confirmed;
em.merge(jpaChange);
} else {
throw new WebApplicationException(String.format("Could not find change with ID: %s", id));
}
} catch (PersistenceException e) {
throw new WebApplicationException(e, Response.serverError().build());
}
}
@Override
@WithRoles
@RolesAllowed(Roles.TESTER)
@Transactional
public void deleteChange(int id) {
if (!ChangeDAO.deleteById(id)) {
throw ServiceException.notFound("Change not found");
}
}
@Override
@RolesAllowed(Roles.TESTER)
@WithRoles
public void recalculateDatapoints(int testId, boolean notify,
boolean debug, Boolean clearDatapoints, Long from, Long to) {
TestDAO test = TestDAO.findById(testId);
if (test == null) {
throw ServiceException.notFound("Test " + testId + " does not exist or is not available.");
} else if (!Roles.hasRoleWithSuffix(identity, test.owner, "-tester")) {
throw ServiceException.forbidden("This user cannot trigger the recalculation");
}
messageBus.executeForTest(testId, () -> {
startRecalculation(testId, notify, debug, clearDatapoints == null ? true : clearDatapoints, from, to);
});
}
void startRecalculation(int testId, boolean notify, boolean debug, boolean clearDatapoints, Long from, Long to) {
Recalculation recalculation = new Recalculation();
Recalculation previous = recalcProgress.putIfAbsent(testId, recalculation);
while (previous != null) {
if (!previous.done) {
log.debugf("Already started recalculation on test %d, ignoring.", testId);
return;
}
if (recalcProgress.replace(testId, previous, recalculation)) {
break;
}
previous = recalcProgress.putIfAbsent(testId, recalculation);
}
recalculation.clearDatapoints = clearDatapoints;
try {
log.debugf("Updating fingerprints for test %d", testId);
//update fingerprints before starting recalculation
//TODO: check if we need to update fingerprints for all tests
mediator.updateFingerprints(testId);
log.debugf("About to recalculate datapoints in test %d between %s and %s", testId, from, to);
//TODO:: determine if we should clear datapoints
recalculation.datasets = getDatasetsForRecalculation(testId, from, to, clearDatapoints);
int numRuns = recalculation.datasets.size();
log.debugf("Starting recalculation of test %d, %d runs", testId, numRuns);
int completed = 0;
recalcProgress.put(testId, recalculation);
//TODO:: this could be more streamlined
Map lastDatapoints = new HashMap<>();
recalculation.datasets.entrySet().forEach(entry -> lastDatapoints.put(entry.getValue(), entry.getKey()));
Set lastDatapointSet = lastDatapoints.values().stream().collect(Collectors.toSet());
for (int datasetId : recalculation.datasets.keySet()) {
// Since the evaluation might take few moments and we're dealing potentially with thousands
// of runs we'll process each run in a separate transaction
if (lastDatapointSet.contains(datasetId)) {
recalculation.lastDatapoint = true;
} else {
recalculation.lastDatapoint = false;
}
recalculateForDataset(datasetId, notify, debug, recalculation);
recalculation.progress = 100 * ++completed / numRuns;
}
} catch (Throwable t) {
log.error("Recalculation failed", t);
throw t;
} finally {
recalculation.done = true;
vertx.setTimer(30_000, timerId -> recalcProgress.remove(testId, recalculation));
}
}
// It doesn't make sense to limit access to particular user when doing the recalculation,
// normally the calculation happens with system privileges anyway.
@WithRoles(extras = Roles.HORREUM_SYSTEM)
@Transactional
Map getDatasetsForRecalculation(Integer testId, Long from, Long to, boolean clearDatapoints) {
Query query = session
.createNativeQuery(
"SELECT id, fingerprint FROM dataset LEFT JOIN fingerprint ON dataset.id = fingerprint.dataset_id WHERE testid = ?1 AND (EXTRACT(EPOCH FROM start) * 1000 BETWEEN ?2 AND ?3) ORDER BY start",
Tuple.class)
.setParameter(1, testId)
.setParameter(2, from == null ? Long.MIN_VALUE : from)
.setParameter(3, to == null ? Long.MAX_VALUE : to);
//use LinkedHashMap to preserve ordering of ID insertion - query is ordered on start
Map ids = new LinkedHashMap();
query.getResultList().forEach(tuple -> ids.put(
((Integer) ((Tuple) tuple).get("id")).intValue(),
((Tuple) tuple).get("fingerprint") == null ? "" : ((String) ((Tuple) tuple).get("fingerprint")).toString()));
List datasetIDs = ids.keySet().stream().collect(Collectors.toList());
if (clearDatapoints) {
DataPointDAO.delete("dataset.id in ?1", datasetIDs);
}
ChangeDAO.delete("dataset.id in ?1 AND confirmed = false", datasetIDs);
if (!ids.isEmpty()) {
// Due to RLS policies we cannot add a record to a dataset we don't own
logCalculationMessage(testId, datasetIDs.get(0), PersistentLogDAO.INFO, "Starting recalculation of %d runs.",
ids.size());
}
return ids;
}
@WithRoles(extras = Roles.HORREUM_SYSTEM)
@Transactional(Transactional.TxType.REQUIRES_NEW)
void recalculateForDataset(Integer datasetId, boolean notify, boolean debug, Recalculation recalculation) {
DatasetDAO dataset = DatasetDAO.findById(datasetId);
if (dataset != null) {
recalculateDatapointsForDataset(dataset, notify, debug, recalculation);
} else {
log.debugf("Could not find dataset with id: %d", datasetId);
}
}
@Override
@RolesAllowed(Roles.TESTER)
public DatapointRecalculationStatus getRecalculationStatus(int testId) {
Recalculation recalculation = recalcProgress.get(testId);
DatapointRecalculationStatus status = new DatapointRecalculationStatus();
status.percentage = recalculation == null ? 100 : recalculation.progress;
status.done = recalculation == null || recalculation.done;
if (recalculation != null) {
status.totalDatasets = recalculation.datasets.size();
status.errors = recalculation.errors;
status.datasetsWithoutValue = recalculation.datasetsWithoutValue.values().stream().map(DatasetMapper::fromInfo)
.collect(Collectors.toList());
}
return status;
}
@WithRoles(extras = Roles.HORREUM_SYSTEM)
@Transactional
@Scheduled(every = "{horreum.alerting.missing.dataset.check}")
public void checkMissingDataset() {
List results = session.createNativeQuery(LOOKUP_RECENT, Object[].class).getResultList();
for (Object[] row : results) {
int ruleId = (int) row[0];
int testId = (int) row[1];
String ruleName = (String) row[2];
long maxStaleness = (long) row[3];
Instant timestamp = (Instant) row[4];
if (timestamp == null || timestamp.isBefore(timeService.now().minusMillis(maxStaleness))) {
if (ruleName == null) {
ruleName = "rule #" + ruleId;
}
notificationService.notifyMissingDataset(testId, ruleName, maxStaleness, timestamp);
int numUpdated = em.createNativeQuery("UPDATE missingdata_rule SET last_notification = ?1 WHERE id = ?2")
.setParameter(1, timeService.now()).setParameter(2, ruleId).executeUpdate();
if (numUpdated != 1) {
log.errorf("Missing data rules update for rule %d (test %d) didn't work: updated: %d", ruleId, testId,
numUpdated);
}
}
}
}
@Override
@WithRoles
@PermitAll
public List findLastDatapoints(LastDatapointsParams params) {
//noinspection unchecked
return em.createNativeQuery(FIND_LAST_DATAPOINTS)
.unwrap(NativeQuery.class)
.setParameter(1, Util.parseFingerprint(params.fingerprint), JsonBinaryType.INSTANCE)
.setParameter(2, params.variables, IntArrayType.INSTANCE)
.setTupleTransformer((tuples, aliases) -> {
return new DatapointLastTimestamp((int) tuples[0], (Number) tuples[1]);
}).getResultList();
}
@Override
@RolesAllowed(Roles.UPLOADER)
@WithRoles
@Transactional
public void expectRun(String testNameOrId, Long timeoutSeconds, String expectedBy, String backlink) {
if (timeoutSeconds == null) {
throw ServiceException.badRequest("No timeout set.");
} else if (timeoutSeconds <= 0) {
throw ServiceException.badRequest("Timeout must be positive (unit: seconds)");
}
TestDAO test = testService.ensureTestExists(testNameOrId);
RunExpectationDAO runExpectation = new RunExpectationDAO();
runExpectation.testId = test.id;
runExpectation.expectedBefore = timeService.now().plusSeconds(timeoutSeconds);
runExpectation.expectedBy = expectedBy != null ? expectedBy : identity.getPrincipal().getName();
runExpectation.backlink = backlink;
runExpectation.persist();
}
@PermitAll
@WithRoles(extras = Roles.HORREUM_SYSTEM)
@Override
public List expectations() {
List expectations = RunExpectationDAO.listAll();
return expectations.stream().map(RunExpectationMapper::from).collect(Collectors.toList());
}
@WithRoles
@Transactional
@Override
public void updateChangeDetection(int testId, AlertingService.ChangeDetectionUpdate update) {
TestDAO test = testService.getTestForUpdate(testId);
test.timelineLabels = toJsonArray(update.timelineLabels);
test.timelineFunction = "";
test.timelineFunction = update.timelineFunction;
test.fingerprintLabels = toJsonArray(update.fingerprintLabels);
// In case the filter is null we need to force the property to be dirty
test.fingerprintFilter = "";
test.fingerprintFilter = update.fingerprintFilter;
test.persistAndFlush();
}
private ArrayNode toJsonArray(List labels) {
if (labels == null) {
return null;
}
return labels.stream().reduce(JsonNodeFactory.instance.arrayNode(), ArrayNode::add, ArrayNode::addAll);
}
@PermitAll
@Override
public List changeDetectionModels() {
return modelResolver.getModels().values().stream().map(ChangeDetectionModel::config).collect(Collectors.toList());
}
@PermitAll
@Override
public List defaultChangeDetectionConfigs() {
ChangeDetectionDAO lastDatapoint = new ChangeDetectionDAO();
lastDatapoint.model = ChangeDetectionModelType.names.RELATIVE_DIFFERENCE;
lastDatapoint.config = JsonNodeFactory.instance.objectNode()
.put("window", 1).put("model", ChangeDetectionModelType.names.RELATIVE_DIFFERENCE).put("filter", "mean")
.put("threshold", 0.2).put("minPrevious", 5);
ChangeDetectionDAO floatingWindow = new ChangeDetectionDAO();
floatingWindow.model = ChangeDetectionModelType.names.RELATIVE_DIFFERENCE;
floatingWindow.config = JsonNodeFactory.instance.objectNode()
.put("window", 5).put("model", ChangeDetectionModelType.names.RELATIVE_DIFFERENCE).put("filter", "mean")
.put("threshold", 0.1).put("minPrevious", 5);
return Arrays.asList(lastDatapoint, floatingWindow).stream().map(ChangeDetectionMapper::from)
.collect(Collectors.toList());
}
@WithRoles
@Override
public List missingDataRules(int testId) {
if (testId <= 0) {
throw ServiceException.badRequest("Invalid test ID: " + testId);
}
List rules = MissingDataRuleDAO.list("test.id", testId);
return rules.stream().map(MissingDataRuleMapper::from).collect(Collectors.toList());
}
@WithRoles
@Transactional
@Override
public int updateMissingDataRule(int testId, MissingDataRule dto) {
MissingDataRuleDAO rule = MissingDataRuleMapper.to(dto);
// check test existence and ownership
testService.getTestForUpdate(testId);
if (rule.id != null && rule.id <= 0) {
rule.id = null;
}
// drop any info about last notification
rule.lastNotification = null;
if (rule.maxStaleness <= 0) {
throw ServiceException.badRequest("Invalid max staleness in rule " + rule.name + ": " + rule.maxStaleness);
}
if (rule.id == null) {
rule.test = em.getReference(TestDAO.class, testId);
rule.persistAndFlush();
} else {
MissingDataRuleDAO existing = MissingDataRuleDAO.findById(rule.id);
if (existing == null) {
throw ServiceException.badRequest("Rule does not exist.");
} else if (existing.test.id != testId) {
throw ServiceException.badRequest("Rule belongs to a different test");
}
rule.test = existing.test;
em.merge(rule);
em.flush();
}
// The recalculations are executed in independent transactions, therefore we need to make sure that
// this rule is committed in DB before starting to reevaluate it.
Util.doAfterCommit(tm, () -> {
messageBus.executeForTest(testId, () -> {
recalculateMissingDataRules(testId, rule);
});
});
return rule.id;
}
@WithRoles(extras = Roles.HORREUM_SYSTEM)
@Transactional
void recalculateMissingDataRules(int testId, MissingDataRuleDAO rule) {
List idsAndTimestamps = session
.createNativeQuery("SELECT id, start FROM dataset WHERE testid = ?1", Object[].class)
.setParameter(1, testId).getResultList();
for (Object[] row : idsAndTimestamps) {
recalculateMissingDataRule((int) row[0], (Instant) row[1], rule);
}
}
@WithRoles(extras = Roles.HORREUM_SYSTEM)
@Transactional(Transactional.TxType.REQUIRES_NEW)
void recalculateMissingDataRule(int datasetId, Instant timestamp, MissingDataRuleDAO rule) {
JsonNode value = (JsonNode) em.createNativeQuery(LOOKUP_LABEL_VALUE_FOR_RULE)
.setParameter(1, datasetId).setParameter(2, rule.id)
.unwrap(NativeQuery.class)
.addScalar("value", JsonBinaryType.INSTANCE)
.getSingleResult();
boolean match = true;
if (rule.condition != null && !rule.condition.isBlank()) {
String ruleName = rule.name == null ? "#" + rule.id : rule.name;
match = Util.evaluateTest(rule.condition, value, notBoolean -> {
logMissingDataMessage(rule.testId(), datasetId, PersistentLogDAO.ERROR,
"Missing data rule %s result is not a boolean: %s", ruleName, notBoolean);
return true;
},
(code, exception) -> logMissingDataMessage(rule.testId(), datasetId, PersistentLogDAO.ERROR,
"Error evaluating missing data rule %s: '%s' Code:%s ", ruleName, exception.getMessage(),
code),
output -> logMissingDataMessage(rule.testId(), datasetId, PersistentLogDAO.DEBUG,
"Output while evaluating missing data rule %s: '%s'", ruleName, output));
}
if (match) {
new MissingDataRuleResultDAO(rule.id, datasetId, timestamp).persist();
}
}
@WithRoles
@Transactional
@Override
public void deleteMissingDataRule(int id) {
MissingDataRuleResultDAO.deleteForDataRule(id);
MissingDataRuleDAO.deleteById(id);
}
@WithRoles(extras = Roles.HORREUM_SYSTEM)
@Transactional
public void removeExpected(Run run) {
// delete at most one expectation
Query query = em.createNativeQuery("""
DELETE FROM run_expectation
WHERE id = (
SELECT id
FROM run_expectation
WHERE testid = (
SELECT testid
FROM run
WHERE id = ?1
)
LIMIT 1)
""");
query.setParameter(1, run.id);
int updated = query.executeUpdate();
if (updated > 0) {
log.debugf("Removed %d run expectations as run %d was added.", updated, run.id);
}
}
@WithRoles(extras = Roles.HORREUM_SYSTEM)
@Transactional
void onDatasetDeleted(int datasetId) {
log.debugf("Removing changes for dataset %d", datasetId);
ChangeDAO.delete("dataset.id = ?1 AND confirmed = false", datasetId);
DataPointDAO.delete("dataset.id", datasetId);
//Need to make sure we delete MissingDataRuleResults when datasets are removed
MissingDataRuleResultDAO.deleteForDataset(datasetId);
}
@WithRoles(extras = Roles.HORREUM_SYSTEM)
@Transactional
void onTestDeleted(int testId) {
// We need to delete in a loop to cascade this to ChangeDetection
List variables = VariableDAO.list("testId", testId);
log.debugf("Deleting %d variables for test (%d)", variables.size(), testId);
for (var variable : variables) {
variable.delete();
}
MissingDataRuleDAO.delete("test.id", testId);
em.flush();
}
@Transactional
@WithRoles(extras = Roles.HORREUM_SYSTEM)
@Scheduled(every = "{horreum.alerting.expected.run.check}")
public void checkExpectedRuns() {
for (RunExpectationDAO expectation : RunExpectationDAO
. find("expectedBefore < ?1", timeService.now()).list()) {
boolean sendNotifications = (Boolean) em.createNativeQuery("SELECT notificationsenabled FROM test WHERE id = ?")
.setParameter(1, expectation.testId).getSingleResult();
if (sendNotifications) {
// We will perform this only if this transaction succeeds, to allow no-op retries
Util.doAfterCommit(tm, () -> notificationService.notifyExpectedRun(expectation.testId,
expectation.expectedBefore.toEpochMilli(), expectation.expectedBy, expectation.backlink));
} else {
log.debugf("Skipping expected run notification on test %d since it is disabled.", expectation.testId);
}
expectation.delete();
}
}
// Note: this class must be public - otherwise when this is used as a parameter to
// a method in AlertingServiceImpl the interceptors would not be invoked.
public static class Recalculation {
Map datasets = Collections.emptyMap();
int progress;
boolean done;
public int errors;
boolean lastDatapoint;
boolean clearDatapoints;
Map datasetsWithoutValue = new HashMap<>();
}
static final class VarAndFingerprint {
final int varId;
final JsonNode fingerprint;
VarAndFingerprint(int varId, JsonNode fingerprint) {
this.varId = varId;
this.fingerprint = fingerprint;
}
@Override
public boolean equals(Object o) {
if (this == o)
return true;
if (o == null || getClass() != o.getClass())
return false;
VarAndFingerprint that = (VarAndFingerprint) o;
return varId == that.varId && Objects.equals(fingerprint, that.fingerprint);
}
@Override
public int hashCode() {
return Objects.hash(varId, fingerprint);
}
}
private static class UpTo {
final Instant timestamp;
final boolean inclusive;
private UpTo(Instant timestamp, boolean inclusive) {
this.timestamp = timestamp;
this.inclusive = inclusive;
}
@Override
public String toString() {
return "{ts=" + timestamp +
", incl=" + inclusive +
'}';
}
}
}