mirror of
https://github.com/MarginaliaSearch/MarginaliaSearch.git
synced 2025-10-06 07:32:38 +02:00
Compare commits
5 Commits
deploy-022
...
deploy-023
Author | SHA1 | Date | |
---|---|---|---|
|
9f041d6631 | ||
|
13fb1efce4 | ||
|
c1225165b7 | ||
|
67ad7a3bbc | ||
|
ed62ec8a35 |
@@ -0,0 +1,6 @@
|
|||||||
|
-- Add additional summary columns to DOMAIN_SECURITY_EVENTS table
|
||||||
|
-- to make it easier to make sense of certificate changes
|
||||||
|
|
||||||
|
ALTER TABLE DOMAIN_SECURITY_EVENTS ADD COLUMN CHANGE_CERTIFICATE_SERIAL_NUMBER BOOLEAN NOT NULL DEFAULT FALSE;
|
||||||
|
ALTER TABLE DOMAIN_SECURITY_EVENTS ADD COLUMN CHANGE_CERTIFICATE_ISSUER BOOLEAN NOT NULL DEFAULT FALSE;
|
||||||
|
OPTIMIZE TABLE DOMAIN_SECURITY_EVENTS;
|
@@ -13,15 +13,12 @@ import nu.marginalia.mq.persistence.MqMessageHandlerRegistry;
|
|||||||
import nu.marginalia.mq.persistence.MqPersistence;
|
import nu.marginalia.mq.persistence.MqPersistence;
|
||||||
import nu.marginalia.mqapi.ProcessInboxNames;
|
import nu.marginalia.mqapi.ProcessInboxNames;
|
||||||
import nu.marginalia.mqapi.ping.PingRequest;
|
import nu.marginalia.mqapi.ping.PingRequest;
|
||||||
import nu.marginalia.nodecfg.NodeConfigurationService;
|
|
||||||
import nu.marginalia.nodecfg.model.NodeProfile;
|
|
||||||
import nu.marginalia.process.ProcessService;
|
import nu.marginalia.process.ProcessService;
|
||||||
import nu.marginalia.service.module.ServiceConfiguration;
|
import nu.marginalia.service.module.ServiceConfiguration;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
import java.sql.SQLException;
|
import java.sql.SQLException;
|
||||||
import java.util.Set;
|
|
||||||
import java.util.concurrent.ExecutionException;
|
import java.util.concurrent.ExecutionException;
|
||||||
import java.util.concurrent.ExecutorService;
|
import java.util.concurrent.ExecutorService;
|
||||||
import java.util.concurrent.Executors;
|
import java.util.concurrent.Executors;
|
||||||
@@ -41,7 +38,6 @@ public class PingMonitorActor extends RecordActorPrototype {
|
|||||||
private final ProcessService.ProcessId processId;
|
private final ProcessService.ProcessId processId;
|
||||||
private final ExecutorService executorService = Executors.newSingleThreadExecutor();
|
private final ExecutorService executorService = Executors.newSingleThreadExecutor();
|
||||||
private final int node;
|
private final int node;
|
||||||
private final boolean isPrimaryNode;
|
|
||||||
private final Gson gson;
|
private final Gson gson;
|
||||||
|
|
||||||
public record Initial() implements ActorStep {}
|
public record Initial() implements ActorStep {}
|
||||||
@@ -56,7 +52,7 @@ public class PingMonitorActor extends RecordActorPrototype {
|
|||||||
public ActorStep transition(ActorStep self) throws Exception {
|
public ActorStep transition(ActorStep self) throws Exception {
|
||||||
return switch (self) {
|
return switch (self) {
|
||||||
case Initial i -> {
|
case Initial i -> {
|
||||||
PingRequest request = new PingRequest(isPrimaryNode ? "primary": "secondary");
|
PingRequest request = new PingRequest();
|
||||||
|
|
||||||
persistence.sendNewMessage(inboxName, null, null,
|
persistence.sendNewMessage(inboxName, null, null,
|
||||||
"PingRequest",
|
"PingRequest",
|
||||||
@@ -129,7 +125,6 @@ public class PingMonitorActor extends RecordActorPrototype {
|
|||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
public PingMonitorActor(Gson gson,
|
public PingMonitorActor(Gson gson,
|
||||||
NodeConfigurationService nodeConfigurationService,
|
|
||||||
ServiceConfiguration configuration,
|
ServiceConfiguration configuration,
|
||||||
MqPersistence persistence,
|
MqPersistence persistence,
|
||||||
ProcessService processService) throws SQLException {
|
ProcessService processService) throws SQLException {
|
||||||
@@ -140,9 +135,6 @@ public class PingMonitorActor extends RecordActorPrototype {
|
|||||||
this.processService = processService;
|
this.processService = processService;
|
||||||
this.inboxName = ProcessInboxNames.PING_INBOX + ":" + node;
|
this.inboxName = ProcessInboxNames.PING_INBOX + ":" + node;
|
||||||
this.processId = ProcessService.ProcessId.PING;
|
this.processId = ProcessService.ProcessId.PING;
|
||||||
|
|
||||||
this.isPrimaryNode = Set.of(NodeProfile.BATCH_CRAWL, NodeProfile.MIXED)
|
|
||||||
.contains(nodeConfigurationService.get(node).profile());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Sets the message to dead in the database to avoid
|
/** Sets the message to dead in the database to avoid
|
||||||
|
@@ -27,10 +27,12 @@ public class DbBrowseDomainsRandom {
|
|||||||
public List<BrowseResult> getRandomDomains(int count, DomainBlacklist blacklist, int set) {
|
public List<BrowseResult> getRandomDomains(int count, DomainBlacklist blacklist, int set) {
|
||||||
|
|
||||||
final String q = """
|
final String q = """
|
||||||
SELECT DOMAIN_ID, DOMAIN_NAME, INDEXED
|
SELECT EC_RANDOM_DOMAINS.DOMAIN_ID, DOMAIN_NAME, INDEXED
|
||||||
FROM EC_RANDOM_DOMAINS
|
FROM EC_RANDOM_DOMAINS
|
||||||
INNER JOIN EC_DOMAIN ON EC_DOMAIN.ID=DOMAIN_ID
|
INNER JOIN EC_DOMAIN ON EC_DOMAIN.ID=DOMAIN_ID
|
||||||
|
LEFT JOIN DOMAIN_AVAILABILITY_INFORMATION DAI ON DAI.DOMAIN_ID=EC_RANDOM_DOMAINS.DOMAIN_ID
|
||||||
WHERE STATE<2
|
WHERE STATE<2
|
||||||
|
AND SERVER_AVAILABLE
|
||||||
AND DOMAIN_SET=?
|
AND DOMAIN_SET=?
|
||||||
AND DOMAIN_ALIAS IS NULL
|
AND DOMAIN_ALIAS IS NULL
|
||||||
ORDER BY RAND()
|
ORDER BY RAND()
|
||||||
|
@@ -10,7 +10,6 @@ import nu.marginalia.mq.MessageQueueFactory;
|
|||||||
import nu.marginalia.mqapi.ProcessInboxNames;
|
import nu.marginalia.mqapi.ProcessInboxNames;
|
||||||
import nu.marginalia.mqapi.ping.PingRequest;
|
import nu.marginalia.mqapi.ping.PingRequest;
|
||||||
import nu.marginalia.nodecfg.NodeConfigurationService;
|
import nu.marginalia.nodecfg.NodeConfigurationService;
|
||||||
import nu.marginalia.nodecfg.model.NodeConfiguration;
|
|
||||||
import nu.marginalia.process.ProcessConfiguration;
|
import nu.marginalia.process.ProcessConfiguration;
|
||||||
import nu.marginalia.process.ProcessConfigurationModule;
|
import nu.marginalia.process.ProcessConfigurationModule;
|
||||||
import nu.marginalia.process.ProcessMainClass;
|
import nu.marginalia.process.ProcessMainClass;
|
||||||
@@ -21,7 +20,6 @@ import org.slf4j.Logger;
|
|||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
import java.security.Security;
|
import java.security.Security;
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
public class PingMain extends ProcessMainClass {
|
public class PingMain extends ProcessMainClass {
|
||||||
private static final Logger log = LoggerFactory.getLogger(PingMain.class);
|
private static final Logger log = LoggerFactory.getLogger(PingMain.class);
|
||||||
@@ -56,56 +54,6 @@ public class PingMain extends ProcessMainClass {
|
|||||||
// Start the ping job scheduler
|
// Start the ping job scheduler
|
||||||
pingJobScheduler.start(true);
|
pingJobScheduler.start(true);
|
||||||
|
|
||||||
// Watch the crawler process to suspend/resume the ping job scheduler
|
|
||||||
try {
|
|
||||||
serviceRegistry.watchProcess("crawler", node, (running) -> {
|
|
||||||
if (running) {
|
|
||||||
log.info("Crawler process is running, suspending ping job scheduler.");
|
|
||||||
pingJobScheduler.pause(node);
|
|
||||||
} else {
|
|
||||||
log.warn("Crawler process is not running, resuming ping job scheduler.");
|
|
||||||
pingJobScheduler.resume(node);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
catch (Exception e) {
|
|
||||||
throw new RuntimeException("Failed to watch crawler process", e);
|
|
||||||
}
|
|
||||||
|
|
||||||
log.info("PingMain started successfully.");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
public void runSecondary() {
|
|
||||||
log.info("Starting PingMain...");
|
|
||||||
|
|
||||||
List<Integer> crawlerNodes = nodeConfigurationService.getAll()
|
|
||||||
.stream()
|
|
||||||
.filter(node -> !node.disabled())
|
|
||||||
.filter(node -> node.profile().permitBatchCrawl())
|
|
||||||
.map(NodeConfiguration::node)
|
|
||||||
.toList()
|
|
||||||
;
|
|
||||||
|
|
||||||
// Start the ping job scheduler
|
|
||||||
pingJobScheduler.start(true);
|
|
||||||
|
|
||||||
// Watch the crawler process to suspend/resume the ping job scheduler
|
|
||||||
try {
|
|
||||||
serviceRegistry.watchProcessAnyNode("crawler", crawlerNodes, (running, n) -> {
|
|
||||||
if (running) {
|
|
||||||
log.info("Crawler process is running on node {} taking over ", n);
|
|
||||||
pingJobScheduler.resume(n);
|
|
||||||
} else {
|
|
||||||
log.warn("Crawler process stopped, resuming ping job scheduler.");
|
|
||||||
pingJobScheduler.pause(n);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
catch (Exception e) {
|
|
||||||
throw new RuntimeException("Failed to watch crawler process", e);
|
|
||||||
}
|
|
||||||
|
|
||||||
log.info("PingMain started successfully.");
|
log.info("PingMain started successfully.");
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -144,19 +92,8 @@ public class PingMain extends ProcessMainClass {
|
|||||||
var instructions = main.fetchInstructions(PingRequest.class);
|
var instructions = main.fetchInstructions(PingRequest.class);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
switch (instructions.value().runClass) {
|
|
||||||
case "primary":
|
|
||||||
log.info("Running as primary node");
|
|
||||||
main.runPrimary();
|
main.runPrimary();
|
||||||
break;
|
for(;;) main.wait(); // Wait on the object lock to avoid busy-looping
|
||||||
case "secondary":
|
|
||||||
log.info("Running as secondary node");
|
|
||||||
main.runSecondary();
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
throw new IllegalArgumentException("Invalid runClass: " + instructions.value().runClass);
|
|
||||||
}
|
|
||||||
for(;;);
|
|
||||||
}
|
}
|
||||||
catch (Throwable ex) {
|
catch (Throwable ex) {
|
||||||
logger.error("Error running ping process", ex);
|
logger.error("Error running ping process", ex);
|
||||||
|
@@ -83,7 +83,7 @@ public class PingHttpFetcher {
|
|||||||
} catch (SocketTimeoutException ex) {
|
} catch (SocketTimeoutException ex) {
|
||||||
return new TimeoutResponse(ex.getMessage());
|
return new TimeoutResponse(ex.getMessage());
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
return new ConnectionError(e.getMessage());
|
return new ConnectionError(e.getClass().getSimpleName());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -16,6 +16,8 @@ public record DomainSecurityEvent(
|
|||||||
boolean certificateProfileChanged,
|
boolean certificateProfileChanged,
|
||||||
boolean certificateSanChanged,
|
boolean certificateSanChanged,
|
||||||
boolean certificatePublicKeyChanged,
|
boolean certificatePublicKeyChanged,
|
||||||
|
boolean certificateSerialNumberChanged,
|
||||||
|
boolean certificateIssuerChanged,
|
||||||
Duration oldCertificateTimeToExpiry,
|
Duration oldCertificateTimeToExpiry,
|
||||||
boolean securityHeadersChanged,
|
boolean securityHeadersChanged,
|
||||||
boolean ipChanged,
|
boolean ipChanged,
|
||||||
@@ -41,8 +43,10 @@ public record DomainSecurityEvent(
|
|||||||
change_software,
|
change_software,
|
||||||
old_cert_time_to_expiry,
|
old_cert_time_to_expiry,
|
||||||
security_signature_before,
|
security_signature_before,
|
||||||
security_signature_after
|
security_signature_after,
|
||||||
) VALUES (?,?,?,?,?,?,?,?,?,?,?,?,?,?)
|
change_certificate_serial_number,
|
||||||
|
change_certificate_issuer
|
||||||
|
) VALUES (?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?)
|
||||||
"""))
|
"""))
|
||||||
{
|
{
|
||||||
|
|
||||||
@@ -75,6 +79,9 @@ public record DomainSecurityEvent(
|
|||||||
ps.setBytes(14, securitySignatureAfter().compressed());
|
ps.setBytes(14, securitySignatureAfter().compressed());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ps.setBoolean(15, certificateSerialNumberChanged());
|
||||||
|
ps.setBoolean(16, certificateIssuerChanged());
|
||||||
|
|
||||||
ps.executeUpdate();
|
ps.executeUpdate();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -15,6 +15,8 @@ public record SecurityInformationChange(
|
|||||||
boolean isCertificateProfileChanged,
|
boolean isCertificateProfileChanged,
|
||||||
boolean isCertificateSanChanged,
|
boolean isCertificateSanChanged,
|
||||||
boolean isCertificatePublicKeyChanged,
|
boolean isCertificatePublicKeyChanged,
|
||||||
|
boolean isCertificateSerialNumberChanged,
|
||||||
|
boolean isCertificateIssuerChanged,
|
||||||
Duration oldCertificateTimeToExpiry,
|
Duration oldCertificateTimeToExpiry,
|
||||||
boolean isSecurityHeadersChanged,
|
boolean isSecurityHeadersChanged,
|
||||||
boolean isIpAddressChanged,
|
boolean isIpAddressChanged,
|
||||||
@@ -30,8 +32,10 @@ public record SecurityInformationChange(
|
|||||||
|
|
||||||
boolean certificateFingerprintChanged = 0 != Arrays.compare(before.sslCertFingerprintSha256(), after.sslCertFingerprintSha256());
|
boolean certificateFingerprintChanged = 0 != Arrays.compare(before.sslCertFingerprintSha256(), after.sslCertFingerprintSha256());
|
||||||
boolean certificateProfileChanged = before.certificateProfileHash() != after.certificateProfileHash();
|
boolean certificateProfileChanged = before.certificateProfileHash() != after.certificateProfileHash();
|
||||||
|
boolean certificateSerialNumberChanged = !Objects.equals(before.sslCertSerialNumber(), after.sslCertSerialNumber());
|
||||||
boolean certificatePublicKeyChanged = 0 != Arrays.compare(before.sslCertPublicKeyHash(), after.sslCertPublicKeyHash());
|
boolean certificatePublicKeyChanged = 0 != Arrays.compare(before.sslCertPublicKeyHash(), after.sslCertPublicKeyHash());
|
||||||
boolean certificateSanChanged = !Objects.equals(before.sslCertSan(), after.sslCertSan());
|
boolean certificateSanChanged = !Objects.equals(before.sslCertSan(), after.sslCertSan());
|
||||||
|
boolean certificateIssuerChanged = !Objects.equals(before.sslCertIssuer(), after.sslCertIssuer());
|
||||||
|
|
||||||
Duration oldCertificateTimeToExpiry = before.sslCertNotAfter() == null ? null : Duration.between(
|
Duration oldCertificateTimeToExpiry = before.sslCertNotAfter() == null ? null : Duration.between(
|
||||||
Instant.now(),
|
Instant.now(),
|
||||||
@@ -50,6 +54,7 @@ public record SecurityInformationChange(
|
|||||||
boolean isChanged = asnChanged
|
boolean isChanged = asnChanged
|
||||||
|| certificateFingerprintChanged
|
|| certificateFingerprintChanged
|
||||||
|| securityHeadersChanged
|
|| securityHeadersChanged
|
||||||
|
|| certificateProfileChanged
|
||||||
|| softwareChanged;
|
|| softwareChanged;
|
||||||
|
|
||||||
return new SecurityInformationChange(
|
return new SecurityInformationChange(
|
||||||
@@ -59,6 +64,8 @@ public record SecurityInformationChange(
|
|||||||
certificateProfileChanged,
|
certificateProfileChanged,
|
||||||
certificateSanChanged,
|
certificateSanChanged,
|
||||||
certificatePublicKeyChanged,
|
certificatePublicKeyChanged,
|
||||||
|
certificateSerialNumberChanged,
|
||||||
|
certificateIssuerChanged,
|
||||||
oldCertificateTimeToExpiry,
|
oldCertificateTimeToExpiry,
|
||||||
securityHeadersChanged,
|
securityHeadersChanged,
|
||||||
ipChanged,
|
ipChanged,
|
||||||
|
@@ -8,6 +8,7 @@ import nu.marginalia.ping.ssl.PKIXValidationResult;
|
|||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import javax.annotation.Nullable;
|
||||||
import java.security.MessageDigest;
|
import java.security.MessageDigest;
|
||||||
import java.security.NoSuchAlgorithmException;
|
import java.security.NoSuchAlgorithmException;
|
||||||
import java.security.cert.CertificateEncodingException;
|
import java.security.cert.CertificateEncodingException;
|
||||||
@@ -21,13 +22,17 @@ public class DomainSecurityInformationFactory {
|
|||||||
private static final Logger logger = LoggerFactory.getLogger(DomainSecurityInformationFactory.class);
|
private static final Logger logger = LoggerFactory.getLogger(DomainSecurityInformationFactory.class);
|
||||||
|
|
||||||
// Vanilla HTTP (not HTTPS) response does not have SSL session information, so we return null
|
// Vanilla HTTP (not HTTPS) response does not have SSL session information, so we return null
|
||||||
public DomainSecurityRecord createHttpSecurityInformation(HttpResponse httpResponse, int domainId, int nodeId) {
|
public DomainSecurityRecord createHttpSecurityInformation(HttpResponse httpResponse,
|
||||||
|
int domainId, int nodeId,
|
||||||
|
@Nullable Integer asn
|
||||||
|
) {
|
||||||
|
|
||||||
var headers = httpResponse.headers();
|
var headers = httpResponse.headers();
|
||||||
|
|
||||||
return DomainSecurityRecord.builder()
|
return DomainSecurityRecord.builder()
|
||||||
.domainId(domainId)
|
.domainId(domainId)
|
||||||
.nodeId(nodeId)
|
.nodeId(nodeId)
|
||||||
|
.asn(asn)
|
||||||
.httpSchema(HttpSchema.HTTP)
|
.httpSchema(HttpSchema.HTTP)
|
||||||
.httpVersion(httpResponse.version())
|
.httpVersion(httpResponse.version())
|
||||||
.headerServer(headers.getFirst("Server"))
|
.headerServer(headers.getFirst("Server"))
|
||||||
@@ -47,7 +52,13 @@ public class DomainSecurityInformationFactory {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// HTTPS response
|
// HTTPS response
|
||||||
public DomainSecurityRecord createHttpsSecurityInformation(HttpsResponse httpResponse, PKIXValidationResult validationResult, int domainId, int nodeId) {
|
public DomainSecurityRecord createHttpsSecurityInformation(
|
||||||
|
HttpsResponse httpResponse,
|
||||||
|
PKIXValidationResult validationResult,
|
||||||
|
int domainId,
|
||||||
|
int nodeId,
|
||||||
|
@Nullable Integer asn
|
||||||
|
) {
|
||||||
|
|
||||||
|
|
||||||
var headers = httpResponse.headers();
|
var headers = httpResponse.headers();
|
||||||
@@ -86,6 +97,7 @@ public class DomainSecurityInformationFactory {
|
|||||||
return DomainSecurityRecord.builder()
|
return DomainSecurityRecord.builder()
|
||||||
.domainId(domainId)
|
.domainId(domainId)
|
||||||
.nodeId(nodeId)
|
.nodeId(nodeId)
|
||||||
|
.asn(asn)
|
||||||
.httpSchema(HttpSchema.HTTPS)
|
.httpSchema(HttpSchema.HTTPS)
|
||||||
.headerServer(headers.getFirst("Server"))
|
.headerServer(headers.getFirst("Server"))
|
||||||
.headerCorsAllowOrigin(headers.getFirst("Access-Control-Allow-Origin"))
|
.headerCorsAllowOrigin(headers.getFirst("Access-Control-Allow-Origin"))
|
||||||
|
@@ -18,6 +18,7 @@ import java.net.InetAddress;
|
|||||||
import java.net.UnknownHostException;
|
import java.net.UnknownHostException;
|
||||||
import java.security.cert.X509Certificate;
|
import java.security.cert.X509Certificate;
|
||||||
import java.sql.SQLException;
|
import java.sql.SQLException;
|
||||||
|
import java.time.Duration;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
@@ -75,8 +76,8 @@ public class HttpPingService {
|
|||||||
|
|
||||||
result = pingHttpFetcher.fetchUrl(url, Method.HEAD, null, null);
|
result = pingHttpFetcher.fetchUrl(url, Method.HEAD, null, null);
|
||||||
|
|
||||||
if (result instanceof HttpsResponse response && response.httpStatus() == 405) {
|
if (result instanceof HttpsResponse response && shouldTryGET(response.httpStatus())) {
|
||||||
// If we get a 405, we try the GET method instead as not all servers support HEAD requests
|
sleep(Duration.ofSeconds(2));
|
||||||
result = pingHttpFetcher.fetchUrl(url, Method.GET, null, null);
|
result = pingHttpFetcher.fetchUrl(url, Method.GET, null, null);
|
||||||
}
|
}
|
||||||
else if (result instanceof ConnectionError) {
|
else if (result instanceof ConnectionError) {
|
||||||
@@ -84,8 +85,8 @@ public class HttpPingService {
|
|||||||
if (!(result2 instanceof ConnectionError)) {
|
if (!(result2 instanceof ConnectionError)) {
|
||||||
result = result2;
|
result = result2;
|
||||||
}
|
}
|
||||||
if (result instanceof HttpResponse response && response.httpStatus() == 405) {
|
if (result instanceof HttpResponse response && shouldTryGET(response.httpStatus())) {
|
||||||
// If we get a 405, we try the GET method instead as not all servers support HEAD requests
|
sleep(Duration.ofSeconds(2));
|
||||||
result = pingHttpFetcher.fetchUrl(alternateUrl, Method.GET, null, null);
|
result = pingHttpFetcher.fetchUrl(alternateUrl, Method.GET, null, null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -116,7 +117,7 @@ public class HttpPingService {
|
|||||||
domainReference.nodeId(),
|
domainReference.nodeId(),
|
||||||
oldPingStatus,
|
oldPingStatus,
|
||||||
ErrorClassification.CONNECTION_ERROR,
|
ErrorClassification.CONNECTION_ERROR,
|
||||||
null);
|
rsp.errorMessage());
|
||||||
newSecurityInformation = null;
|
newSecurityInformation = null;
|
||||||
}
|
}
|
||||||
case TimeoutResponse rsp -> {
|
case TimeoutResponse rsp -> {
|
||||||
@@ -148,7 +149,8 @@ public class HttpPingService {
|
|||||||
newSecurityInformation = domainSecurityInformationFactory.createHttpSecurityInformation(
|
newSecurityInformation = domainSecurityInformationFactory.createHttpSecurityInformation(
|
||||||
httpResponse,
|
httpResponse,
|
||||||
domainReference.domainId(),
|
domainReference.domainId(),
|
||||||
domainReference.nodeId()
|
domainReference.nodeId(),
|
||||||
|
newPingStatus.asn()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
case HttpsResponse httpsResponse -> {
|
case HttpsResponse httpsResponse -> {
|
||||||
@@ -166,7 +168,8 @@ public class HttpPingService {
|
|||||||
httpsResponse,
|
httpsResponse,
|
||||||
validationResult,
|
validationResult,
|
||||||
domainReference.domainId(),
|
domainReference.domainId(),
|
||||||
domainReference.nodeId()
|
domainReference.nodeId(),
|
||||||
|
newPingStatus.asn()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -190,6 +193,29 @@ public class HttpPingService {
|
|||||||
return generatedRecords;
|
return generatedRecords;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private boolean shouldTryGET(int statusCode) {
|
||||||
|
if (statusCode < 400) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (statusCode == 429) { // Too many requests, we should not retry with GET
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// For all other status codes, we can try a GET request, as many severs do not
|
||||||
|
// cope with HEAD requests properly.
|
||||||
|
|
||||||
|
return statusCode < 600;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void sleep(Duration duration) {
|
||||||
|
try {
|
||||||
|
Thread.sleep(duration.toMillis());
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
Thread.currentThread().interrupt(); // Restore the interrupted status
|
||||||
|
logger.warn("Sleep interrupted", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private void comparePingStatuses(List<WritableModel> generatedRecords,
|
private void comparePingStatuses(List<WritableModel> generatedRecords,
|
||||||
DomainAvailabilityRecord oldPingStatus,
|
DomainAvailabilityRecord oldPingStatus,
|
||||||
DomainAvailabilityRecord newPingStatus) {
|
DomainAvailabilityRecord newPingStatus) {
|
||||||
@@ -258,6 +284,8 @@ public class HttpPingService {
|
|||||||
change.isCertificateProfileChanged(),
|
change.isCertificateProfileChanged(),
|
||||||
change.isCertificateSanChanged(),
|
change.isCertificateSanChanged(),
|
||||||
change.isCertificatePublicKeyChanged(),
|
change.isCertificatePublicKeyChanged(),
|
||||||
|
change.isCertificateSerialNumberChanged(),
|
||||||
|
change.isCertificateIssuerChanged(),
|
||||||
change.oldCertificateTimeToExpiry(),
|
change.oldCertificateTimeToExpiry(),
|
||||||
change.isSecurityHeadersChanged(),
|
change.isSecurityHeadersChanged(),
|
||||||
change.isIpAddressChanged(),
|
change.isIpAddressChanged(),
|
||||||
|
@@ -318,6 +318,8 @@ class PingDaoTest {
|
|||||||
true,
|
true,
|
||||||
false,
|
false,
|
||||||
true,
|
true,
|
||||||
|
true,
|
||||||
|
false,
|
||||||
Duration.ofDays(30),
|
Duration.ofDays(30),
|
||||||
false,
|
false,
|
||||||
false,
|
false,
|
||||||
|
@@ -1,9 +1,8 @@
|
|||||||
package nu.marginalia.mqapi.ping;
|
package nu.marginalia.mqapi.ping;
|
||||||
|
|
||||||
public class PingRequest {
|
public class PingRequest {
|
||||||
public final String runClass;
|
|
||||||
|
|
||||||
public PingRequest(String runClass) {
|
public PingRequest() {
|
||||||
this.runClass = runClass;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user