org.openlca.proto.io.server.DataFetchService Maven / Gradle / Ivy
package org.openlca.proto.io.server;
import java.util.Comparator;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
import java.util.function.Consumer;
import java.util.function.Function;
import com.google.protobuf.Empty;
import org.openlca.core.database.CategoryDao;
import org.openlca.core.database.Daos;
import org.openlca.core.database.IDatabase;
import org.openlca.core.matrix.cache.ProcessTable;
import org.openlca.core.matrix.index.TechFlow;
import org.openlca.core.matrix.linking.ProviderIndex;
import org.openlca.core.model.Category;
import org.openlca.core.model.Flow;
import org.openlca.core.model.FlowType;
import org.openlca.core.model.RefEntity;
import org.openlca.core.model.descriptors.RootDescriptor;
import org.openlca.core.model.descriptors.Descriptor;
import org.openlca.core.model.descriptors.ProcessDescriptor;
import org.openlca.proto.ProtoRef;
import org.openlca.proto.grpc.CategoryTree;
import org.openlca.proto.grpc.DataFetchServiceGrpc;
import org.openlca.proto.grpc.FindRequest;
import org.openlca.proto.grpc.GetAllRequest;
import org.openlca.proto.grpc.GetAllResponse;
import org.openlca.proto.grpc.GetCategoryContentRequest;
import org.openlca.proto.grpc.GetCategoryTreeRequest;
import org.openlca.proto.grpc.GetDescriptorsRequest;
import org.openlca.proto.grpc.GetRequest;
import org.openlca.proto.grpc.ProtoDataSet;
import org.openlca.proto.grpc.ProtoTechFlow;
import org.openlca.proto.grpc.SearchRequest;
import org.openlca.proto.io.output.Refs;
import org.openlca.util.Strings;
import io.grpc.stub.StreamObserver;
class DataFetchService extends
DataFetchServiceGrpc.DataFetchServiceImplBase {
private final IDatabase db;
DataFetchService(IDatabase db) {
this.db = db;
}
@Override
public void get(GetRequest req, StreamObserver resp) {
var modelType = DataUtil.forceRootTypeOf(req.getType(), resp);
if (modelType == null)
return;
var id = req.getId();
if (Strings.nullOrEmpty(id)) {
Response.invalidArg(resp, "A data set ID is required");
return;
}
var model = db.get(modelType.getModelClass(), id);
if (model == null) {
Response.notFound(resp,
"No " + modelType + " with ID=" + id + " exists");
return;
}
resp.onNext(DataUtil.toDataSet(db, model).build());
resp.onCompleted();
}
@Override
public void find(FindRequest req, StreamObserver resp) {
var modelType = DataUtil.forceRootTypeOf(req.getType(), resp);
if (modelType == null)
return;
var type = modelType.getModelClass();
Consumer onSuccess = model -> {
resp.onNext(model == null
? ProtoDataSet.newBuilder().build()
: DataUtil.toDataSet(db, model).build());
resp.onCompleted();
};
var id = req.getId();
if (Strings.notEmpty(id)) {
onSuccess.accept(db.get(type, id));
return;
}
var name = req.getName();
onSuccess.accept(Strings.notEmpty(name)
? db.getForName(type, name)
: null);
}
@Override
public void getAll(GetAllRequest req, StreamObserver resp) {
var modelType = DataUtil.forceRootTypeOf(req.getType(), resp);
if (modelType == null)
return;
// first get all descriptors and sort them by ID so that
// we have a defined order for the pages
var type = modelType.getModelClass();
var all = db.getDescriptors(type);
all.sort(Comparator.comparingLong(d -> d.id));
var totalCount = all.size();
var response = GetAllResponse.newBuilder()
.setTotalCount(totalCount);
// select the page
List extends Descriptor> selected;
if (req.getSkipPaging()) {
response.setPage(1);
selected = all;
} else {
var pageSize = req.getPageSize() > 0
? req.getPageSize()
: 100;
var page = req.getPage() > 0
? req.getPage()
: 1;
response.setPage(page);
var offset = (page - 1) * pageSize;
if (offset >= totalCount) {
resp.onNext(response.build());
resp.onCompleted();
return;
}
var end = Math.min(totalCount, offset + pageSize);
selected = all.subList(offset, end);
}
response.setPageSize(selected.size());
selected.stream()
.map(d -> db.get(type, d.id))
.map(e -> DataUtil.toDataSet(db, e))
.forEach(response::addDataSet);
resp.onNext(response.build());
resp.onCompleted();
}
@Override
public void getDescriptors(
GetDescriptorsRequest req, StreamObserver resp) {
var modelType = DataUtil.forceRootTypeOf(req.getType(), resp);
if (modelType == null)
return;
var dao = Daos.root(db, modelType);
if (dao == null) {
resp.onCompleted();
return;
}
var refData = Refs.dataOf(db);
// get by id
var id = req.getId();
if (Strings.notEmpty(id)) {
var d = dao.getDescriptorForRefId(id);
if (d != null) {
resp.onNext(Refs.refOf(d, refData).build());
}
resp.onCompleted();
return;
}
var stream = dao.getDescriptors().stream();
// filter by category
var catId = req.hasAttributes()
? req.getAttributes().getCategory()
: null;
if (Strings.notEmpty(catId)) {
stream = stream.filter(Objects::nonNull);
// "/" identifies the root category or no
// specific category
if (catId.equals("/")) {
stream = stream.filter(d -> {
var cd = (RootDescriptor) d;
return cd.category == null;
});
} else {
var category = DataUtil.getCategory(
db, modelType, catId);
if (category == null) {
resp.onCompleted();
return;
}
stream = stream.filter(d -> {
var cd = (RootDescriptor) d;
return cd.category != null
&& cd.category == category.id;
});
}
}
// filter by name
var name = req.hasAttributes()
? req.getAttributes().getName()
: null;
if (Strings.notEmpty(name)) {
var term = name.trim().toLowerCase();
stream = stream.filter(d -> {
if (d.name == null)
return false;
var other = d.name.trim().toLowerCase();
return other.equals(term);
});
}
stream.forEach(d -> resp.onNext(
Refs.refOf(d, refData).build()));
resp.onCompleted();
}
@Override
public void search(SearchRequest req, StreamObserver resp) {
var refData = Refs.dataOf(db);
Search.of(db, req).run()
.map(descriptor -> Refs.refOf(descriptor, refData))
.forEach(ref -> resp.onNext(ref.build()));
resp.onCompleted();
}
@Override
public void getCategoryContent(
GetCategoryContentRequest req, StreamObserver resp) {
var modelType = DataUtil.forceCategorizedTypeOf(req.getType(), resp);
if (modelType == null)
return;
// find the category
Optional category;
var catID = req.getCategory();
if (Strings.nullOrEmpty(catID) || catID.equals("/")) {
category = Optional.empty();
} else {
var cat = DataUtil.getCategory(db, modelType, catID);
if (cat == null) {
Response.notFound(resp,
"Category with ID ot path =" + catID + " does not exists");
return;
}
category = Optional.of(cat);
}
var dao = Daos.root(db, modelType);
for (var d : dao.getDescriptors(category)) {
resp.onNext(Refs.refOf(d).build());
}
resp.onCompleted();
}
@Override
public void getCategoryTree(
GetCategoryTreeRequest req, StreamObserver resp) {
var modelType = DataUtil.forceCategorizedTypeOf(req.getType(), resp);
if (modelType == null)
return;
var root = CategoryTree.newBuilder();
root.setType(req.getType());
new CategoryDao(db).getRootCategories(modelType)
.forEach(c -> expand(root, c));
resp.onNext(root.build());
resp.onCompleted();
}
private void expand(CategoryTree.Builder parent, Category category) {
var tree = CategoryTree.newBuilder()
.setType(parent.getType())
.setId(Strings.orEmpty(category.refId))
.setName(Strings.orEmpty(category.name));
category.childCategories.forEach(c -> expand(tree, c));
parent.addSubTree(tree);
}
@Override
public void getProvidersFor(ProtoRef req, StreamObserver resp) {
var flow = DataUtil.model(db, Flow.class)
.forRef(req)
.reportErrorsOn(resp)
.get()
.orElse(null);
if (flow == null)
return;
if (flow.flowType == FlowType.ELEMENTARY_FLOW) {
resp.onCompleted();
return;
}
var refData = Refs.dataOf(db);
ProviderIndex.lazy(db)
.getProvidersOf(flow.id)
.stream()
.map(TechFlow::provider)
.filter(p -> p instanceof ProcessDescriptor)
.map(p -> (ProcessDescriptor) p)
.forEach(p -> resp.onNext(Refs.refOf(p, refData).build()));
resp.onCompleted();
}
@Override
public void getTechFlows(Empty req, StreamObserver resp) {
var refData = Refs.dataOf(db);
Function mapper = techFlow -> {
var flow = techFlow.flow();
var process = techFlow.provider();
if (flow == null || process == null)
return null;
var proto = ProtoTechFlow.newBuilder()
.setProvider(Refs.refOf(process, refData));
var protoFlow = Refs.refOf(flow, refData);
if (flow.flowType == FlowType.WASTE_FLOW) {
proto.setWaste(protoFlow);
} else {
proto.setProduct(protoFlow);
}
return proto.build();
};
ProcessTable.create(db).getProviders()
.stream()
.map(mapper)
.filter(Objects::nonNull)
.forEach(resp::onNext);
resp.onCompleted();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy