С самого начала я всегда путался с тем, как бороться с InterruptedException и как правильно отменить HTTP-запрос, если они занимают слишком много времени. У меня есть библиотека, в которой я предоставил два метода, синхронизацию и асинхронность для нашего клиента. Они могут вызывать тот метод, который, по их мнению, подходит для их целей.Как отменить HTTP-запрос AsyncRestTemplate, если они занимают слишком много времени?
- executeSync() - ждет, пока у меня есть результат, возвращает результат.
- executeAsync() - немедленно возвращает Будущее, которое может быть обработано после выполнения других действий, если это необходимо.
Они передадут объект DataKey
, который имеет идентификатор пользователя и значение таймаута в нем. Мы выясним, какую машину вызывать на основе идентификатора пользователя, а затем создать URL-адрес с этой машиной, и мы сделаем http-обращение к URL-адресу, используя AsyncRestTemplate, а затем отправьте ответ на них по вопросу о том, успешна она или нет.
Я использую exchange метод AsyncRestTemplate
который возвращает обратно ListenableFuture
, и я хотел бы иметь асинхронную ноны архитектуры блокировки с клиентскими подключениями на основе НИО так, что запрос использует не блокирует IO так, поэтому я пошел с AsyncRestTemplate
. Подходит ли этот подход к определению моей проблемы? Эта библиотека будет использоваться в производстве при очень большой нагрузке.
Ниже мой интерфейс:
public interface Client {
// for synchronous
public DataResponse executeSync(DataKey key);
// for asynchronous
public ListenableFuture<DataResponse> executeAsync(DataKey key);
}
И ниже моя реализация интерфейса:
public class DataClient implements Client {
// using spring 4 AsyncRestTemplate
private final AsyncRestTemplate restTemplate = new AsyncRestTemplate();
// for synchronous
@Override
public DataResponse executeSync(DataKey keys) {
Future<DataResponse> responseFuture = executeAsync(keys);
DataResponse response = null;
try {
response = responseFuture.get(keys.getTimeout(), TimeUnit.MILLISECONDS);
} catch (InterruptedException ex) {
// do we need to catch InterruptedException here and interrupt the thread?
Thread.currentThread().interrupt();
// also do I need throw this RuntimeException at all?
throw new RuntimeException("Interrupted", ex);
} catch (TimeoutException ex) {
DataLogging.logEvents(ex, DataErrorEnum.CLIENT_TIMEOUT, keys);
response = new DataResponse(null, DataErrorEnum.CLIENT_TIMEOUT, DataStatusEnum.ERROR);
responseFuture.cancel(true); // terminating the tasks that got timed out so that they don't take up the resources?
} catch (Exception ex) {
DataLogging.logEvents(ex, DataErrorEnum.ERROR_CLIENT, keys);
response = new DataResponse(null, DataErrorEnum.ERROR_CLIENT, DataStatusEnum.ERROR);
}
return response;
}
// for asynchronous
@Override
public ListenableFuture<DataResponse> executeAsync(final DataKey keys) {
final SettableFuture<DataResponse> responseFuture = SettableFuture.create();
final org.springframework.util.concurrent.ListenableFuture orig =
restTemplate.exchange(createURL(keys), HttpMethod.GET, keys.getEntity(), String.class);
orig.addCallback(
new ListenableFutureCallback<ResponseEntity<String>>() {
@Override
public void onSuccess(ResponseEntity<String> result) {
responseFuture.set(new DataResponse(result.getBody(), DataErrorEnum.OK,
DataStatusEnum.SUCCESS));
}
@Override
public void onFailure(Throwable ex) {
DataLogging.logErrors(ex, DataErrorEnum.ERROR_SERVER, keys);
responseFuture.set(new DataResponse(null, DataErrorEnum.ERROR_SERVER,
DataStatusEnum.ERROR));
}
});
// propagate cancellation back to the original request
responseFuture.addListener(new Runnable() {
@Override public void run() {
if (responseFuture.isCancelled()) {
orig.cancel(false); // I am keeping this false for now
}
}
}, MoreExecutors.directExecutor());
return responseFuture;
}
}
И клиент будет вызывающему как это от своего кода -
// if they are calling executeSync() method
DataResponse response = DataClientFactory.getInstance().executeSync(dataKey);
// and if they want to call executeAsync() method
Future<DataResponse> response = DataClientFactory.getInstance().executeAsync(dataKey);
Теперь задан вопрос -
Можем ли мы прервать вызов
AsyncRestTemplate
, если HTTP-запрос занимает слишком много времени? Я на самом деле звонюcancel
по моемуfuture
в моем приведенном выше коде в методеexecuteSync
, но я не уверен, как его проверить, чтобы убедиться, что он делает то, что должен? Я хочу отменить отмену в исходное будущее, чтобы отменить соответствующий HTTP-запрос (который я, вероятно, хочу сделать для сохранения ресурсов), поэтому я добавил слушателя в свой метод executeAsync. Я считаю, мы не можем прерывать звонкиRestTemplate
, но не уверены в том, можем ли мы это сделать или нет. Если предположим, что мы можем прерватьAsyncRestTemplate
звонки, то я делаю все правильно, чтобы прервать http-звонки? Или есть лучший/более чистый способ сделать это? Или мне даже нужно беспокоиться об отмене запроса Http с помощьюAsyncRestTemplate
с моим текущим дизайном?// propagate cancellation back to the original request responseFuture.addListener(new Runnable() { @Override public void run() { if (responseFuture.isCancelled()) { orig.cancel(false); // I am keeping this false for now } } }, MoreExecutors.directExecutor());
С текущей настройки, я могу видеть, что это бросает CancellationException часть времени (не каждый раз) - Означает ли это, что мой запрос HTTP был отменен тогда?
- Также я делаю правильную вещь в блоке catch
InterruptedException
вexecuteSync
методе? Если нет, то каков правильный способ справиться с этим. И мне нужно иметь дело сInterruptedException
вообще в моем случае? - Верно ли, что по умолчанию
AsyncRestTamplete
использует блокирующие вызовы и запросы на поток? Если да, то есть ли способ подключения клиентов на основе NIO в моей текущей настройке?
Любые объяснения/предложения по коду будут очень полезны.
Я не могу ответить на все ваши вопросы, но могу сказать, что вам не нужно прерывать текущий поток в блоке catch 'InterruptedException'. Возможно, вы захотите добавить значение 'CLIENT_INTERRUPTED' в свой' DataErrorEnum' и вернуть ответ об ошибке, аналогичный вашим другим блокам catch. –
Я не могу ответить на все ваши вопросы, но мне было интересно, почему вы не используете тайм-ауты сокетов для этих целей, передаются ли данные по строке и вы хотите отменить соединение? –