io.trino.server.remotetask.SimpleHttpResponseHandler Maven / Gradle / Ivy
/*
* Licensed 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 io.trino.server.remotetask;
import com.google.common.util.concurrent.FutureCallback;
import io.airlift.http.client.FullJsonResponseHandler;
import io.airlift.http.client.HttpStatus;
import io.trino.spi.TrinoException;
import java.net.URI;
import static io.trino.spi.StandardErrorCode.REMOTE_TASK_ERROR;
import static java.lang.String.format;
import static java.util.Objects.requireNonNull;
public class SimpleHttpResponseHandler
implements FutureCallback>
{
private final SimpleHttpResponseCallback callback;
private final URI uri;
private final RemoteTaskStats stats;
public SimpleHttpResponseHandler(SimpleHttpResponseCallback callback, URI uri, RemoteTaskStats stats)
{
this.callback = callback;
this.uri = uri;
this.stats = requireNonNull(stats, "stats is null");
}
@Override
public void onSuccess(FullJsonResponseHandler.JsonResponse response)
{
stats.updateSuccess();
stats.responseSize(response.getResponseSize());
try {
if (response.getStatusCode() == HttpStatus.OK.code() && response.hasValue()) {
callback.success(response.getValue());
}
else if (response.getStatusCode() == HttpStatus.SERVICE_UNAVAILABLE.code()) {
callback.failed(new ServiceUnavailableException(uri));
}
else {
// Something is broken in the server or the client, so fail the task immediately (includes 500 errors)
Exception cause = response.getException();
if (cause == null) {
if (response.getStatusCode() == HttpStatus.OK.code()) {
cause = new TrinoException(REMOTE_TASK_ERROR, format("Expected response from %s is empty", uri));
}
else {
cause = new TrinoException(REMOTE_TASK_ERROR, format("Expected response code from %s to be %s, but was %s%n%s",
uri,
HttpStatus.OK.code(),
response.getStatusCode(),
response.getResponseBody()));
}
}
else {
cause = new TrinoException(REMOTE_TASK_ERROR, format("Unexpected response from %s", uri), cause);
}
callback.fatal(cause);
}
}
catch (Throwable t) {
// this should never happen
callback.fatal(t);
}
}
@Override
public void onFailure(Throwable t)
{
stats.updateFailure();
callback.failed(t);
}
private static class ServiceUnavailableException
extends RuntimeException
{
public ServiceUnavailableException(URI uri)
{
super("Server returned SERVICE_UNAVAILABLE: " + uri);
}
}
}