package io.vertx.reactivex.impl;
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.ObjectCodec;
import com.fasterxml.jackson.core.type.TypeReference;
import io.reactivex.Flowable;
import io.reactivex.FlowableTransformer;
import io.reactivex.Maybe;
import io.reactivex.Single;
import io.reactivex.annotations.NonNull;
import io.vertx.core.buffer.Buffer;
import org.reactivestreams.Publisher;
import static io.vertx.reactivex.impl.ObservableUnmarshaller.getT;
import static java.util.Objects.nonNull;
public class FlowableUnmarshaller<T, B> implements FlowableTransformer<B, T> {
private final java.util.function.Function<B, Buffer> unwrap;
private final Class<T> mappedType;
private final TypeReference<T> mappedTypeRef;
private final ObjectCodec mapper;
public FlowableUnmarshaller(java.util.function.Function<B, Buffer> unwrap, Class<T> mappedType) {
this(unwrap, mappedType, null, null);
}
public FlowableUnmarshaller(java.util.function.Function<B, Buffer> unwrap, TypeReference<T> mappedTypeRef) {
this(unwrap, null, mappedTypeRef, null);
}
public FlowableUnmarshaller(java.util.function.Function<B, Buffer> unwrap, Class<T> mappedType, ObjectCodec mapper) {
this(unwrap, mappedType, null, mapper);
}
public FlowableUnmarshaller(java.util.function.Function<B, Buffer> unwrap, TypeReference<T> mappedTypeRef, ObjectCodec mapper) {
this(unwrap, null, mappedTypeRef, mapper);
}
private FlowableUnmarshaller(java.util.function.Function<B, Buffer> unwrap, Class<T> mappedType, TypeReference<T> mappedTypeRef, ObjectCodec mapper) {
this.unwrap = unwrap;
this.mappedType = mappedType;
this.mappedTypeRef = mappedTypeRef;
this.mapper = mapper;
}
@Override
public Publisher<T> apply(@NonNull Flowable<B> upstream) {
Flowable<Buffer> unwrapped = upstream.map(unwrap::apply);
Single<Buffer> aggregated = unwrapped.collect(Buffer::buffer, Buffer::appendBuffer);
Maybe<T> unmarshalled = aggregated.toMaybe().concatMap(buffer -> {
if (buffer.length() > 0) {
try {
T obj;
if (mapper != null) {
JsonParser parser = mapper.getFactory().createParser(buffer.getBytes());
obj = nonNull(mappedType) ? mapper.readValue(parser, mappedType) :
mapper.readValue(parser, mappedTypeRef);
} else {
obj = getT(buffer, mappedType, mappedTypeRef);
}
return Maybe.just(obj);
} catch (Exception e) {
return Maybe.error(e);
}
} else {
return Maybe.empty();
}
});
return unmarshalled.toFlowable();
}
}