org.killbill.queue.retry.RetryableService Maven / Gradle / Ivy
/*
* Copyright 2014-2017 Groupon, Inc
* Copyright 2014-2017 The Billing Project, LLC
*
* The Billing Project licenses this file to you under the Apache License, version 2.0
* (the "License"); you may not use this file except in compliance with the
* License. You may obtain a copy of the License at:
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*/
package org.killbill.queue.retry;
import java.io.IOException;
import java.util.List;
import java.util.UUID;
import org.joda.time.DateTime;
import org.joda.time.Period;
import org.killbill.billing.util.queue.QueueRetryException;
import org.killbill.notificationq.api.NotificationEvent;
import org.killbill.notificationq.api.NotificationQueue;
import org.killbill.notificationq.api.NotificationQueueService;
import org.killbill.notificationq.api.NotificationQueueService.NoSuchNotificationQueue;
import org.killbill.notificationq.api.NotificationQueueService.NotificationQueueAlreadyExists;
import org.killbill.notificationq.api.NotificationQueueService.NotificationQueueHandler;
import org.killbill.queue.QueueObjectMapper;
import org.killbill.queue.api.QueueEvent;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.fasterxml.jackson.databind.ObjectMapper;
public abstract class RetryableService {
public static final String RETRYABLE_SERVICE_NAME = "notifications-retries";
private static final Logger log = LoggerFactory.getLogger(RetryableService.class);
private final ObjectMapper objectMapper;
private final NotificationQueueService notificationQueueService;
private NotificationQueue retryNotificationQueue;
public RetryableService(NotificationQueueService notificationQueueService) {
this(notificationQueueService, QueueObjectMapper.get());
}
public RetryableService(final NotificationQueueService notificationQueueService,
final ObjectMapper objectMapper) {
this.notificationQueueService = notificationQueueService;
this.objectMapper = objectMapper;
}
public void initialize(final NotificationQueue originalQueue, final NotificationQueueHandler originalQueueHandler) {
initialize(originalQueue.getQueueName(), originalQueueHandler);
}
public void initialize(final String queueName, final NotificationQueueHandler originalQueueHandler) {
try {
final NotificationQueueHandler notificationQueueHandler = new NotificationQueueHandler() {
@Override
public void handleReadyNotification(final NotificationEvent eventJson,
final DateTime eventDateTime,
final UUID userToken,
final Long searchKey1,
final Long searchKey2) {
if (eventJson instanceof RetryNotificationEvent) {
final RetryNotificationEvent retryNotificationEvent = (RetryNotificationEvent) eventJson;
final NotificationEvent notificationEvent;
try {
notificationEvent = (NotificationEvent) objectMapper.readValue(retryNotificationEvent.getOriginalEvent(), retryNotificationEvent.getOriginalEventClass());
} catch (final IOException e) {
throw new RuntimeException(e);
}
try {
originalQueueHandler.handleReadyNotification(notificationEvent,
eventDateTime,
userToken,
searchKey1,
searchKey2);
} catch (final QueueRetryException e) {
scheduleRetry(e,
notificationEvent,
retryNotificationEvent.getOriginalEffectiveDate(),
userToken,
searchKey1,
searchKey2,
retryNotificationEvent.getRetryNb() + 1);
}
} else {
log.error("Retry service received an unexpected event className='{}'", eventJson.getClass());
}
}
};
this.retryNotificationQueue = notificationQueueService.createNotificationQueue(RETRYABLE_SERVICE_NAME,
queueName,
notificationQueueHandler);
} catch (final NotificationQueueAlreadyExists notificationQueueAlreadyExists) {
throw new RuntimeException(notificationQueueAlreadyExists);
}
}
public void start() {
retryNotificationQueue.startQueue();
}
public void stop() throws NoSuchNotificationQueue {
if (retryNotificationQueue != null) {
retryNotificationQueue.stopQueue();
notificationQueueService.deleteNotificationQueue(retryNotificationQueue.getServiceName(), retryNotificationQueue.getQueueName());
}
}
public void scheduleRetry(final QueueRetryException exception,
final QueueEvent originalNotificationEvent,
final DateTime originalEffectiveDate,
final UUID userToken,
final Long searchKey1,
final Long searchKey2,
final int retryNb) {
final DateTime effectiveDate = computeRetryDate(exception, originalEffectiveDate, retryNb);
if (effectiveDate == null) {
log.warn("Error processing event, NOT scheduling retry for event='{}', retryNb='{}'", originalNotificationEvent, retryNb, exception);
throw new RetryableInternalException(false);
}
log.warn("Error processing event, scheduling retry for event='{}', effectiveDate='{}', retryNb='{}'", originalNotificationEvent, effectiveDate, retryNb, exception);
try {
final NotificationEvent retryNotificationEvent = new RetryNotificationEvent(objectMapper.writeValueAsString(originalNotificationEvent), originalNotificationEvent.getClass(), originalEffectiveDate, retryNb);
retryNotificationQueue.recordFutureNotification(effectiveDate, retryNotificationEvent, userToken, searchKey1, searchKey2);
throw new RetryableInternalException(true);
} catch (final IOException e) {
log.error("Unable to schedule retry for event='{}', effectiveDate='{}'", originalNotificationEvent, effectiveDate, e);
throw new RetryableInternalException(false);
}
}
private DateTime computeRetryDate(final QueueRetryException queueRetryException, final DateTime initialEventDateTime, final int retryNb) {
final List retrySchedule = queueRetryException.getRetrySchedule();
if (retrySchedule == null || retryNb > retrySchedule.size()) {
return null;
} else {
final Period nextDelay = retrySchedule.get(retryNb - 1);
return initialEventDateTime.plus(nextDelay);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy