/*
* Copyright (C) 2012-2016 Facebook, Inc.
*
* 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.facebook.nifty.processor;
import com.facebook.nifty.core.RequestContext;
import com.google.common.util.concurrent.Futures;
import com.google.common.util.concurrent.ListenableFuture;
import org.apache.thrift.TException;
import org.apache.thrift.TProcessor;
import org.apache.thrift.TProcessorFactory;
import org.apache.thrift.protocol.TProtocol;
import org.apache.thrift.transport.TTransport;
import java.util.concurrent.ExecutionException;
public class NiftyProcessorAdapters
{
/**
* Adapt a {@link TProcessor} to a standard Thrift {@link NiftyProcessor}. Nifty uses this
* internally to adapt the processors generated by the standard Thrift code generator into
* instances of {@link NiftyProcessor} usable by {@link com.facebook.nifty.core.NiftyDispatcher}
*/
public static NiftyProcessor processorFromTProcessor(final TProcessor standardThriftProcessor)
{
checkProcessMethodSignature();
return new NiftyProcessor()
{
@Override
public ListenableFuture<Boolean> process(TProtocol in, TProtocol out, RequestContext requestContext) throws TException
{
return Futures.immediateFuture(standardThriftProcessor.process(in, out));
}
};
}
/**
* Create a {@link NiftyProcessorFactory} that always returns the same {@link NiftyProcessor}
* adapted from the given standard Thrift {@link TProcessor}
*/
public static NiftyProcessorFactory factoryFromTProcessor(final TProcessor standardThriftProcessor)
{
checkProcessMethodSignature();
return new NiftyProcessorFactory()
{
@Override
public NiftyProcessor getProcessor(TTransport transport)
{
return processorFromTProcessor(standardThriftProcessor);
}
};
}
/**
* Create a {@link NiftyProcessorFactory} that delegates to a standard Thrift {@link TProcessorFactory}
* to construct an instance, then adapts each instance to a {@link NiftyProcessor}
*/
public static NiftyProcessorFactory factoryFromTProcessorFactory(final TProcessorFactory standardThriftProcessorFactory)
{
checkProcessMethodSignature();
return new NiftyProcessorFactory()
{
@Override
public NiftyProcessor getProcessor(TTransport transport)
{
return processorFromTProcessor(standardThriftProcessorFactory.getProcessor
(transport));
}
};
}
/**
* Adapt a {@link NiftyProcessor} to a standard Thrift {@link TProcessor}. The {@link
* com.facebook.nifty.core.NiftyRequestContext} will always be {@code null}
*/
public static TProcessor processorToTProcessor(final NiftyProcessor niftyProcessor)
{
return new TProcessor()
{
@Override
public boolean process(TProtocol in, TProtocol out) throws TException
{
try {
return niftyProcessor.process(in, out, null).get();
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
throw new TException(e);
} catch (ExecutionException e) {
throw new TException(e);
}
}
};
}
/**
* Create a standard thrift {@link TProcessorFactory} that always returns the same
* {@link TProcessor} adapted from the given {@link NiftyProcessor}
*/
public static TProcessorFactory processorToTProcessorFactory(final NiftyProcessor niftyProcessor)
{
return new TProcessorFactory(processorToTProcessor(niftyProcessor));
}
/**
* Create a standard thrift {@link TProcessorFactory} that delegates to a
* {@link NiftyProcessorFactory} to construct an instance, then adapts each instance to a
* standard Thrift {@link TProcessor}
*/
public static TProcessorFactory processorFactoryToTProcessorFactory(final NiftyProcessorFactory niftyProcessorFactory)
{
return new TProcessorFactory(null) {
@Override
public TProcessor getProcessor(TTransport trans)
{
return processorToTProcessor(niftyProcessorFactory.getProcessor(trans));
}
};
}
/**
* Catch the mismatch early if someone tries to pass our internal variant of TProcessor with
* a different signature on the process() method into these adapters.
*/
private static void checkProcessMethodSignature()
{
try {
TProcessor.class.getMethod("process", TProtocol.class, TProtocol.class);
}
catch (NoSuchMethodException e) {
// Facebook's TProcessor variant needs processor adapters from a different package
throw new IllegalStateException("The loaded TProcessor class is not supported by version of the adapters");
}
}
}