org.finos.tracdap.common.codec.arrow.ArrowDecoder Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of tracdap-lib-data Show documentation
Show all versions of tracdap-lib-data Show documentation
TRAC D.A.P. data library, interfaces and core functionality for working with primary data
/*
* Copyright 2022 Accenture Global Solutions Limited
*
* 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 org.finos.tracdap.common.codec.arrow;
import org.finos.tracdap.common.codec.BufferDecoder;
import org.finos.tracdap.common.data.DataPipeline;
import org.finos.tracdap.common.data.util.Bytes;
import org.finos.tracdap.common.exception.EDataCorruption;
import org.finos.tracdap.common.exception.EUnexpected;
import org.apache.arrow.memory.ArrowBuf;
import org.apache.arrow.vector.VectorSchemaRoot;
import org.apache.arrow.vector.ipc.ArrowReader;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.IOException;
import java.util.List;
import java.util.concurrent.Callable;
public abstract class ArrowDecoder extends BufferDecoder implements DataPipeline.BufferApi {
private final Logger log = LoggerFactory.getLogger(getClass());
private List buffer;
private ArrowReader reader;
private VectorSchemaRoot root;
public ArrowDecoder() {
}
protected abstract ArrowReader createReader(List buffer) throws IOException;
@Override
public void onBuffer(List buffer) {
if (log.isTraceEnabled())
log.trace("ARROW DECODER: onBuffer()");
if (Bytes.readableBytes(buffer) == 0) {
var error = new EDataCorruption("Arrow data file is empty");
log.error(error.getMessage(), error);
throw error;
}
handleErrors(() -> {
this.buffer = buffer;
this.reader = createReader(buffer);
this.root = reader.getVectorSchemaRoot();
consumer().onStart(root);
var isComplete = sendBatches();
if (isComplete) {
markAsDone();
consumer().onComplete();
close();
}
return null;
});
}
@Override
public void onError(Throwable error) {
try {
if (log.isTraceEnabled())
log.trace("ARROW DECODER: onError()");
markAsDone();
consumer().onError(error);
}
finally {
close();
}
}
@Override
public void pump() {
handleErrors(() -> {
// Don't try to pump if the stage isn't active yet
if (root == null)
return null;
var isComplete = sendBatches();
if (isComplete) {
markAsDone();
consumer().onComplete();
close();
}
return null;
});
}
private boolean sendBatches() throws IOException {
// Data arrives in one big buffer, so don't send it all at once
// PUsh what the consumer requests, then wait for another call to pump()
while (consumerReady()) {
var batchAvailable = reader.loadNextBatch();
if (batchAvailable) {
consumer().onBatch();
}
else {
return true;
}
}
return false;
}
private void handleErrors(Callable lambda) {
try {
lambda.call();
}
catch (Throwable e) {
var error = ArrowErrorMapping.mapDecodingError(e);
log.error(error.getMessage(), error);
throw error;
}
}
@Override
public void close() {
try {
if (root != null) {
root.close();
root = null;
}
if (reader != null) {
reader.close();
reader = null;
}
if (buffer != null) {
buffer.forEach(ArrowBuf::close);
buffer = null;
}
}
catch (Exception e) {
log.error("Unexpected error while shutting down Arrow file decoder", e);
throw new EUnexpected(e);
}
}
}