2024-10-14 22:34:41 +01:00
|
|
|
package stirling.software.SPDF.service;
|
|
|
|
|
|
|
|
import java.util.HashMap;
|
|
|
|
import java.util.Map;
|
|
|
|
import java.util.concurrent.ConcurrentHashMap;
|
|
|
|
|
2025-03-20 09:39:57 +00:00
|
|
|
import org.slf4j.Logger;
|
|
|
|
import org.slf4j.LoggerFactory;
|
2024-10-14 22:34:41 +01:00
|
|
|
import org.springframework.scheduling.annotation.Scheduled;
|
|
|
|
import org.springframework.stereotype.Service;
|
|
|
|
|
|
|
|
import io.micrometer.core.instrument.MeterRegistry;
|
|
|
|
import io.micrometer.core.instrument.search.Search;
|
|
|
|
|
2025-04-25 15:35:12 +02:00
|
|
|
import lombok.RequiredArgsConstructor;
|
|
|
|
|
2025-03-20 09:39:57 +00:00
|
|
|
import stirling.software.SPDF.config.EndpointInspector;
|
|
|
|
|
2024-10-14 22:34:41 +01:00
|
|
|
@Service
|
2025-04-25 15:35:12 +02:00
|
|
|
@RequiredArgsConstructor
|
2024-10-14 22:34:41 +01:00
|
|
|
public class MetricsAggregatorService {
|
2025-03-20 09:39:57 +00:00
|
|
|
private static final Logger logger = LoggerFactory.getLogger(MetricsAggregatorService.class);
|
2024-10-14 22:34:41 +01:00
|
|
|
|
|
|
|
private final MeterRegistry meterRegistry;
|
|
|
|
private final PostHogService postHogService;
|
2025-03-20 09:39:57 +00:00
|
|
|
private final EndpointInspector endpointInspector;
|
2024-10-14 22:34:41 +01:00
|
|
|
private final Map<String, Double> lastSentMetrics = new ConcurrentHashMap<>();
|
|
|
|
|
2024-12-03 15:26:40 +00:00
|
|
|
@Scheduled(fixedRate = 7200000) // Run every 2 hours
|
2024-10-14 22:34:41 +01:00
|
|
|
public void aggregateAndSendMetrics() {
|
|
|
|
Map<String, Object> metrics = new HashMap<>();
|
2025-03-20 09:39:57 +00:00
|
|
|
|
|
|
|
final boolean validateGetEndpoints = endpointInspector.getValidGetEndpoints().size() != 0;
|
2024-10-14 22:34:41 +01:00
|
|
|
Search.in(meterRegistry)
|
|
|
|
.name("http.requests")
|
|
|
|
.counters()
|
|
|
|
.forEach(
|
|
|
|
counter -> {
|
2024-11-20 08:20:01 +00:00
|
|
|
String method = counter.getId().getTag("method");
|
|
|
|
String uri = counter.getId().getTag("uri");
|
|
|
|
// Skip if either method or uri is null
|
|
|
|
if (method == null || uri == null) {
|
|
|
|
return;
|
|
|
|
}
|
2025-03-20 09:39:57 +00:00
|
|
|
|
|
|
|
// Skip URIs that are 2 characters or shorter
|
|
|
|
if (uri.length() <= 2) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Skip non-GET and non-POST requests
|
2025-01-22 10:39:47 +00:00
|
|
|
if (!"GET".equals(method) && !"POST".equals(method)) {
|
2024-11-26 21:30:47 +00:00
|
|
|
return;
|
|
|
|
}
|
2025-03-20 09:39:57 +00:00
|
|
|
|
|
|
|
// For POST requests, only include if they start with /api/v1
|
|
|
|
if ("POST".equals(method) && !uri.contains("api/v1")) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (uri.contains(".txt")) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// For GET requests, validate if we have a list of valid endpoints
|
|
|
|
if ("GET".equals(method)
|
|
|
|
&& validateGetEndpoints
|
|
|
|
&& !endpointInspector.isValidGetEndpoint(uri)) {
|
|
|
|
logger.debug("Skipping invalid GET endpoint: {}", uri);
|
2024-11-26 21:30:47 +00:00
|
|
|
return;
|
|
|
|
}
|
2024-11-26 20:50:35 +00:00
|
|
|
|
|
|
|
String key =
|
|
|
|
String.format(
|
|
|
|
"http_requests_%s_%s", method, uri.replace("/", "_"));
|
2024-10-14 22:34:41 +01:00
|
|
|
double currentCount = counter.count();
|
|
|
|
double lastCount = lastSentMetrics.getOrDefault(key, 0.0);
|
|
|
|
double difference = currentCount - lastCount;
|
|
|
|
if (difference > 0) {
|
2025-03-20 09:39:57 +00:00
|
|
|
logger.info("{}, {}", key, difference);
|
2024-10-14 22:34:41 +01:00
|
|
|
metrics.put(key, difference);
|
|
|
|
lastSentMetrics.put(key, currentCount);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
// Send aggregated metrics to PostHog
|
|
|
|
if (!metrics.isEmpty()) {
|
2025-03-20 09:39:57 +00:00
|
|
|
|
2024-10-14 22:34:41 +01:00
|
|
|
postHogService.captureEvent("aggregated_metrics", metrics);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|