[openuv] Enhance server side error handling (#12958)
* Enhance server side error handling * Enhancing exception tree * a small factorization of error status * Shorten statusMessage * Correcting log syntax Signed-off-by: clinique <gael@lhopital.org>
This commit is contained in:
parent
822c27c181
commit
3a632ca659
|
@ -59,6 +59,7 @@ import com.google.gson.JsonSyntaxException;
|
||||||
@NonNullByDefault
|
@NonNullByDefault
|
||||||
public class OpenUVBridgeHandler extends BaseBridgeHandler {
|
public class OpenUVBridgeHandler extends BaseBridgeHandler {
|
||||||
private static final String QUERY_URL = "https://api.openuv.io/api/v1/uv?lat=%s&lng=%s&alt=%s";
|
private static final String QUERY_URL = "https://api.openuv.io/api/v1/uv?lat=%s&lng=%s&alt=%s";
|
||||||
|
private static final int RECONNECT_DELAY_MIN = 5;
|
||||||
private static final int REQUEST_TIMEOUT_MS = (int) TimeUnit.SECONDS.toMillis(30);
|
private static final int REQUEST_TIMEOUT_MS = (int) TimeUnit.SECONDS.toMillis(30);
|
||||||
|
|
||||||
private final Logger logger = LoggerFactory.getLogger(OpenUVBridgeHandler.class);
|
private final Logger logger = LoggerFactory.getLogger(OpenUVBridgeHandler.class);
|
||||||
|
@ -69,6 +70,7 @@ public class OpenUVBridgeHandler extends BaseBridgeHandler {
|
||||||
private final LocaleProvider localeProvider;
|
private final LocaleProvider localeProvider;
|
||||||
|
|
||||||
private Optional<ScheduledFuture<?>> reconnectJob = Optional.empty();
|
private Optional<ScheduledFuture<?>> reconnectJob = Optional.empty();
|
||||||
|
private boolean keyVerified;
|
||||||
|
|
||||||
public OpenUVBridgeHandler(Bridge bridge, LocationProvider locationProvider, TranslationProvider i18nProvider,
|
public OpenUVBridgeHandler(Bridge bridge, LocationProvider locationProvider, TranslationProvider i18nProvider,
|
||||||
LocaleProvider localeProvider, Gson gson) {
|
LocaleProvider localeProvider, Gson gson) {
|
||||||
|
@ -82,6 +84,7 @@ public class OpenUVBridgeHandler extends BaseBridgeHandler {
|
||||||
@Override
|
@Override
|
||||||
public void initialize() {
|
public void initialize() {
|
||||||
logger.debug("Initializing OpenUV API bridge handler.");
|
logger.debug("Initializing OpenUV API bridge handler.");
|
||||||
|
keyVerified = false;
|
||||||
BridgeConfiguration config = getConfigAs(BridgeConfiguration.class);
|
BridgeConfiguration config = getConfigAs(BridgeConfiguration.class);
|
||||||
if (config.apikey.isEmpty()) {
|
if (config.apikey.isEmpty()) {
|
||||||
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
|
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
|
||||||
|
@ -94,8 +97,7 @@ public class OpenUVBridgeHandler extends BaseBridgeHandler {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void dispose() {
|
public void dispose() {
|
||||||
reconnectJob.ifPresent(job -> job.cancel(true));
|
freeReconnectJob();
|
||||||
reconnectJob = Optional.empty();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -113,6 +115,8 @@ public class OpenUVBridgeHandler extends BaseBridgeHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
public @Nullable OpenUVResult getUVData(String latitude, String longitude, String altitude) {
|
public @Nullable OpenUVResult getUVData(String latitude, String longitude, String altitude) {
|
||||||
|
String statusMessage = "";
|
||||||
|
ThingStatusDetail statusDetail = ThingStatusDetail.COMMUNICATION_ERROR;
|
||||||
String url = String.format(QUERY_URL, latitude, longitude, altitude);
|
String url = String.format(QUERY_URL, latitude, longitude, altitude);
|
||||||
String jsonData = "";
|
String jsonData = "";
|
||||||
try {
|
try {
|
||||||
|
@ -122,33 +126,53 @@ public class OpenUVBridgeHandler extends BaseBridgeHandler {
|
||||||
String error = uvResponse.getError();
|
String error = uvResponse.getError();
|
||||||
if (error == null) {
|
if (error == null) {
|
||||||
updateStatus(ThingStatus.ONLINE);
|
updateStatus(ThingStatus.ONLINE);
|
||||||
|
keyVerified = true;
|
||||||
return uvResponse.getResult();
|
return uvResponse.getResult();
|
||||||
}
|
}
|
||||||
throw new OpenUVException(error);
|
throw new OpenUVException(error);
|
||||||
}
|
}
|
||||||
} catch (JsonSyntaxException e) {
|
} catch (JsonSyntaxException e) {
|
||||||
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR,
|
if (jsonData.contains("MongoError")) {
|
||||||
String.format("Invalid json received when calling `%s` : %s", url, jsonData));
|
statusMessage = String.format("@text/offline.comm-error-faultly-service [ \"%d\" ]",
|
||||||
|
RECONNECT_DELAY_MIN);
|
||||||
|
scheduleReconnectJob(RECONNECT_DELAY_MIN);
|
||||||
|
} else {
|
||||||
|
statusDetail = ThingStatusDetail.NONE;
|
||||||
|
statusMessage = String.format("@text/offline.invalid-json [ \"%s\" ]", url);
|
||||||
|
logger.debug("{} : {}", statusMessage, jsonData);
|
||||||
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR, e.getMessage());
|
statusMessage = e.getMessage();
|
||||||
} catch (OpenUVException e) {
|
} catch (OpenUVException e) {
|
||||||
if (e.isQuotaError()) {
|
if (e.isQuotaError()) {
|
||||||
LocalDateTime tomorrowMidnight = LocalDate.now().plusDays(1).atStartOfDay().plusMinutes(2);
|
LocalDateTime nextMidnight = LocalDate.now().plusDays(1).atStartOfDay().plusMinutes(2);
|
||||||
|
statusMessage = String.format("@text/offline.comm-error-quota-exceeded [ \"%s\" ]",
|
||||||
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR, String
|
nextMidnight.toString());
|
||||||
.format("@text/offline.comm-error-quota-exceeded [ \"%s\" ]", tomorrowMidnight.toString()));
|
scheduleReconnectJob(Duration.between(LocalDateTime.now(), nextMidnight).toMinutes());
|
||||||
|
} else if (e.isApiKeyError()) {
|
||||||
reconnectJob = Optional.of(scheduler.schedule(this::initiateConnexion,
|
if (keyVerified) {
|
||||||
Duration.between(LocalDateTime.now(), tomorrowMidnight).toMinutes(), TimeUnit.MINUTES));
|
statusMessage = String.format("@text/offline.api-key-not-recognized [ \"%d\" ]",
|
||||||
} else {
|
RECONNECT_DELAY_MIN);
|
||||||
updateStatus(ThingStatus.OFFLINE,
|
scheduleReconnectJob(RECONNECT_DELAY_MIN);
|
||||||
e.isApiKeyError() ? ThingStatusDetail.CONFIGURATION_ERROR : ThingStatusDetail.NONE,
|
} else {
|
||||||
e.getMessage());
|
statusDetail = ThingStatusDetail.CONFIGURATION_ERROR;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
updateStatus(ThingStatus.OFFLINE, statusDetail, statusMessage);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void scheduleReconnectJob(long delay) {
|
||||||
|
freeReconnectJob();
|
||||||
|
reconnectJob = Optional.of(scheduler.schedule(this::initiateConnexion, delay, TimeUnit.MINUTES));
|
||||||
|
}
|
||||||
|
|
||||||
|
private void freeReconnectJob() {
|
||||||
|
reconnectJob.ifPresent(job -> job.cancel(true));
|
||||||
|
reconnectJob = Optional.empty();
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Collection<Class<? extends ThingHandlerService>> getServices() {
|
public Collection<Class<? extends ThingHandlerService>> getServices() {
|
||||||
return Set.of(OpenUVDiscoveryService.class);
|
return Set.of(OpenUVDiscoveryService.class);
|
||||||
|
|
|
@ -66,6 +66,9 @@ channel-type.config.openuv.SafeExposure.index.option.VI = Black
|
||||||
offline.config-error-unknown-apikey = Parameter 'apikey' must be configured.
|
offline.config-error-unknown-apikey = Parameter 'apikey' must be configured.
|
||||||
offline.config-error-invalid-refresh = Parameter 'refresh' must be higher than 3 minutes to stay in free API plan.
|
offline.config-error-invalid-refresh = Parameter 'refresh' must be higher than 3 minutes to stay in free API plan.
|
||||||
offline.comm-error-quota-exceeded = Quota Exceeded, going OFFLINE for today, will retry at : {0}
|
offline.comm-error-quota-exceeded = Quota Exceeded, going OFFLINE for today, will retry at : {0}
|
||||||
|
offline.comm-error-faultly-service = Service not responding, will reconnect in {0} minutes
|
||||||
|
offline.invalid-json = Invalid JSON received when calling `{0}`
|
||||||
|
offline.api-key-not-recognized = Service error while API key is known correct, will reconnect in {0} minutes
|
||||||
|
|
||||||
# discovery result
|
# discovery result
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue