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

io.hyperfoil.tools.horreum.notification.EmailPlugin Maven / Gradle / Ivy

The newest version!
package io.hyperfoil.tools.horreum.notification;

import static java.nio.charset.StandardCharsets.UTF_8;

import java.net.URLEncoder;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.time.Duration;
import java.time.Instant;
import java.time.temporal.ChronoUnit;
import java.util.Date;

import jakarta.enterprise.context.ApplicationScoped;
import jakarta.inject.Inject;

import org.eclipse.microprofile.config.inject.ConfigProperty;

import io.hyperfoil.tools.horreum.events.DatasetChanges;
import io.hyperfoil.tools.horreum.svc.MissingValuesEvent;
import io.hyperfoil.tools.horreum.svc.ServiceException;
import io.quarkus.logging.Log;
import io.quarkus.mailer.Mail;
import io.quarkus.mailer.reactive.ReactiveMailer;
import io.quarkus.qute.Location;
import io.quarkus.qute.Template;

@ApplicationScoped
public class EmailPlugin implements NotificationPlugin {
    @ConfigProperty(name = "horreum.url")
    String baseUrl;

    @ConfigProperty(name = "horreum.mail.subject.prefix", defaultValue = "[Horreum]")
    String subjectPrefix;

    @ConfigProperty(name = "horreum.mail.timeout", defaultValue = "15s")
    Duration sendMailTimeout;

    @Location("change_notification_email")
    Template changeNotificationEmail;

    @Location("missing_dataset_notification_email")
    Template missingDatasetNotificationEmail;

    @Location("missing_values_notification_email")
    Template missingValuesNotificationEmail;

    @Location("expected_run_notification_email")
    Template expectedRunNotificationEmail;

    @Location("api_key_expiration_email")
    Template apiKeyExpirationEmail;

    @Inject
    ReactiveMailer mailer;

    private final DateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

    @Override
    public String method() {
        return "email";
    }

    @Override
    public Notification create(String username, String data) {
        return new EmailNotification(username, data);
    }

    @Override
    public void test(String data) {
        if (data == null || data.isBlank() || !data.contains("@")) {
            throw ServiceException.badRequest(
                    "Mail notifications require an email as a data parameter: '" + data + "' is not a valid email.");
        }
        mailer.send(Mail.withText(data, "Test message", "This is a test message from Horreum. Please ignore.")).await()
                .atMost(sendMailTimeout);
    }

    public class EmailNotification extends Notification {
        protected EmailNotification(String username, String data) {
            super(username, data);
        }

        @Override
        public void notifyChanges(DatasetChanges event) {
            String subject = subjectPrefix + " Change in " + event.testName;
            changeNotificationEmail
                    .data("username", username)
                    .data("testName", event.testName)
                    .data("fingerprint", URLEncoder.encode(event.fingerprint != null ? event.fingerprint : "", UTF_8))
                    .data("baseUrl", baseUrl)
                    .data("testId", String.valueOf(event.dataset.testId))
                    .data("runId", event.dataset.runId)
                    .data("datasetOrdinal", event.dataset.ordinal)
                    .data("changes", event.changes())
                    .createUni().subscribe().with(content -> {
                        mailer.send(Mail.withHtml(data, subject, content)).await().atMost(sendMailTimeout);
                        Log.debug("Sending mail: " + content);
                    });
        }

        @Override
        public void notifyMissingDataset(String testName, int testId, String ruleName, long maxStaleness,
                Instant lastTimestamp) {
            String subject = String.format("%s Missing expected data for %s/%s", subjectPrefix, testName, ruleName);
            missingDatasetNotificationEmail
                    .data("username", username)
                    .data("testName", testName)
                    .data("testId", String.valueOf(testId))
                    .data("ruleName", ruleName)
                    .data("baseUrl", baseUrl)
                    .data("maxStaleness", prettyPrintTime(maxStaleness))
                    .data("currentStaleness",
                            lastTimestamp == null ? "yet"
                                    : "in " + prettyPrintTime(System.currentTimeMillis() - lastTimestamp.toEpochMilli()))
                    .data("lastTimestamp", lastTimestamp == null ? null : dateFormat.format(Date.from(lastTimestamp)))
                    .createUni().subscribe().with(content -> {
                        mailer.send(Mail.withHtml(data, subject, content)).await().atMost(sendMailTimeout);
                        Log.debug("Sending mail: " + content);
                    });
        }

        @Override
        public void notifyMissingValues(String testName, String fingerprint, MissingValuesEvent event) {
            String subject = String.format("%s Missing change detection values in test %s, dataset %d#%d",
                    subjectPrefix, testName, event.dataset.runId, event.dataset.ordinal);
            missingValuesNotificationEmail
                    .data("username", username)
                    .data("testName", testName)
                    .data("testId", String.valueOf(event.dataset.testId))
                    .data("fingerprint", fingerprint)
                    .data("baseUrl", baseUrl)
                    .data("runId", event.dataset.runId)
                    .data("datasetOrdinal", event.dataset.ordinal)
                    .data("variables", event.variables)
                    .createUni().subscribe().with(content -> {
                        mailer.send(Mail.withHtml(data, subject, content)).await().atMost(sendMailTimeout);
                        Log.debug("Sending mail: " + content);
                    });
        }

        @Override
        public void notifyExpectedRun(String testName, int testId, long before, String expectedBy, String backlink) {
            String subject = subjectPrefix + " Missing expected run for " + testName;
            expectedRunNotificationEmail
                    .data("username", username)
                    .data("testName", testName)
                    .data("testId", String.valueOf(testId))
                    .data("baseUrl", baseUrl)
                    .data("before", dateFormat.format(new Date(before)))
                    .data("expectedBy", expectedBy)
                    .data("backlink", backlink)
                    .createUni().subscribe().with(content -> {
                        mailer.send(Mail.withHtml(data, subject, content)).await().atMost(sendMailTimeout);
                        Log.debug("Sending mail: " + content);
                    });
        }

        @Override
        public void notifyApiKeyExpiration(String keyName, Instant creation, Instant lastAccess, long toExpiration,
                long active) {
            String subject = String.format("%s API key \"%s\" %s", subjectPrefix, keyName,
                    toExpiration == -1 ? "EXPIRED" : "about to expire");
            String content = apiKeyExpirationEmail
                    .data("baseUrl", baseUrl)
                    .data("username", username)
                    .data("keyName", keyName)
                    .data("creation", creation.truncatedTo(ChronoUnit.DAYS))
                    .data("lastAccess", lastAccess.truncatedTo(ChronoUnit.DAYS))
                    .data("expiration", toExpiration)
                    .data("active", active)
                    .render();
            mailer.send(Mail.withHtml(data, subject, content)).await().atMost(sendMailTimeout);
            Log.debug("Sending mail: " + content);
        }
    }

    private String prettyPrintTime(long duration) {
        StringBuilder sb = new StringBuilder();
        if (duration >= 86_400_000) {
            long days = duration / 86_400_000;
            sb.append(days).append(" day").append(days > 1 ? "s " : " ");
            duration -= days * 86_400_000;
        }
        if (duration >= 3_600_000) {
            long hours = duration / 3_600_000;
            sb.append(hours).append(" hour").append(hours > 1 ? "s " : " ");
            duration -= hours * 3_600_000;
        }
        if (duration >= 60_000) {
            long minutes = duration / 60_000;
            sb.append(minutes).append(" minute").append(minutes > 1 ? "s " : " ");
            duration -= minutes * 60_000;
        }
        if (duration > 0) {
            sb.append(duration / 1000).append(" second").append(duration >= 2000 ? "s" : "");
            // ignore ms
        }
        return sb.toString().trim();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy