/* Copyright (c) 2015 LinkedIn Corp. 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 com.linkedin.restli.server.multiplexer; import com.linkedin.common.callback.Callback; import com.linkedin.parseq.BaseTask; import com.linkedin.parseq.Context; import com.linkedin.parseq.promise.Promise; import com.linkedin.parseq.promise.Promises; import com.linkedin.parseq.promise.SettablePromise; import com.linkedin.r2.message.RequestContext; import com.linkedin.r2.message.rest.RestException; import com.linkedin.r2.message.rest.RestRequest; import com.linkedin.r2.message.rest.RestResponse; import com.linkedin.r2.transport.common.RestRequestHandler; import com.linkedin.restli.internal.server.RestLiMethodInvoker; /** * A task responsible for executing individual requests. It converts callback based interface of {@link RestRequestHandler} * into a promise based interface. * * If request handling failed with a RestException, it will not fail the request. Instead, it will just returned the encapsulated RestResponse object. * The task can fail immediately if the previous task failed. Any other unexpected exception can also cause the task to fail. * * @author Dmitriy Yefremov */ /* package private */ final class RequestHandlingTask extends BaseTask<RestResponse> { private final RestRequestHandler _requestHandler; private final BaseTask<RestRequest> _request; private final RequestContext _requestContext; private final MultiplexerRunMode _multiplexerRunMode; /* package private */ RequestHandlingTask(RestRequestHandler requestHandler, BaseTask<RestRequest> request, RequestContext requestContext, MultiplexerRunMode multiplexerRunMode) { _requestHandler = requestHandler; _request = request; _requestContext = requestContext; _multiplexerRunMode = multiplexerRunMode; } @Override protected Promise<RestResponse> run(Context context) throws Throwable { final SettablePromise<RestResponse> promise = Promises.settable(); Callback<RestResponse> callback = new Callback<RestResponse>() { @Override public void onError(Throwable e) { // It is assumed that the handler always returns RestException. It happens because the original callback passed into // the handler is wrapped into com.linkedin.restli.internal.server.RestLiCallback. // It is possible the callback is not wrapped and other exceptions are returned // (e.g. com.linkedin.restli.server.RestLiServiceException). This is considered either a programmers mistake or a // runtime exception. if (e instanceof RestException) { promise.done(((RestException) e).getResponse()); } else { promise.fail(e); } } @Override public void onSuccess(RestResponse result) { promise.done(result); } }; if (_request.isFailed()) { callback.onError(_request.getError()); } else { // try invoking the handler try { if (_multiplexerRunMode == MultiplexerRunMode.SINGLE_PLAN) { RestLiMethodInvoker.TASK_CONTEXT.set(context); } _requestHandler.handleRequest(_request.get(), _requestContext, callback); } catch (Exception e) { callback.onError(e); } finally { if (_multiplexerRunMode == MultiplexerRunMode.SINGLE_PLAN) { RestLiMethodInvoker.TASK_CONTEXT.set(null); } } } return promise; } }