package org.simantics.district.network.techtype; import java.io.IOException; import java.nio.charset.Charset; import java.nio.file.Files; import java.nio.file.Paths; import java.util.Collection; import java.util.HashMap; import java.util.Map; import java.util.Objects; import java.util.stream.Collectors; import org.simantics.Simantics; import org.simantics.databoard.Bindings; import org.simantics.databoard.binding.NumberBinding; import org.simantics.databoard.binding.error.BindingException; import org.simantics.databoard.type.Datatype; import org.simantics.databoard.type.NumberType; import org.simantics.databoard.type.StringType; import org.simantics.db.ReadGraph; import org.simantics.db.RequestProcessor; import org.simantics.db.Resource; import org.simantics.db.WriteGraph; import org.simantics.db.common.procedure.adapter.TransientCacheListener; import org.simantics.db.common.request.PossibleIndexRoot; import org.simantics.db.common.request.WriteRequest; import org.simantics.db.exception.DatabaseException; import org.simantics.db.layer0.QueryIndexUtils; import org.simantics.db.layer0.request.PossibleVariable; import org.simantics.db.layer0.request.PropertyInfo; import org.simantics.db.layer0.request.PropertyInfoRequest; import org.simantics.db.layer0.variable.Variable; import org.simantics.district.network.ontology.DistrictNetworkResource; import org.simantics.district.network.techtype.requests.PossibleTechTypeItem; import org.simantics.district.network.techtype.requests.PossibleTechTypeTable; import org.simantics.district.network.techtype.requests.TechTypeTableData; import org.simantics.district.network.techtype.requests.TechTypeTableKeyName; import org.simantics.district.network.techtype.requests.WriteTechTypeTable; import org.simantics.layer0.Layer0; import org.simantics.scl.runtime.SCLContext; import org.simantics.structural.stubs.StructuralResource2; import org.slf4j.Logger; import org.slf4j.LoggerFactory; public class TechTypeUtils { final static Logger LOGGER = LoggerFactory.getLogger(TechTypeUtils.class); public static String DEFAULT_KEY_NAME = "pipeCode"; /** * Load a tech type table from a CSV file and write it in the active model. * * @param componentType The component type that the tech type table is associated with * @param filePath A path to a CSV file * @throws DatabaseException * @throws IOException */ public static void loadTechTypeTable(Resource componentType, String filePath) throws DatabaseException, IOException { String data; try { data = Files.lines(Paths.get(filePath), Charset.defaultCharset()).collect(Collectors.joining("\n")); } catch (IOException e) { LOGGER.error("Failed to read contents of file '{}' as {}", filePath, Charset.defaultCharset(), e); return; } Simantics.getSession().syncRequest(new WriteRequest() { @Override public void perform(WriteGraph graph) throws DatabaseException { graph.syncRequest(new WriteTechTypeTable(componentType, data)); } }); } /** * Execute a cached query for a possible tech type table item in a table with a given item key. * * Result is null, if no such item was found. * * @param session A request processor on which to run the query * @param itemCode A key value, such as a pipe code * @throws DatabaseException */ public static Map getTableItem(RequestProcessor session, Resource table, String itemCode) throws DatabaseException { return session.syncRequest(new PossibleTechTypeItem(table, itemCode), TransientCacheListener.instance()); } /** * Get a single row in a tech type table for a given item code * * Result is null, if no such item was found. * * @param table A TechTypeTable resource * @param itemCode A key value, such as a pipe code * @return A map from property name to value * @throws DatabaseException */ public static Map getTableItem(Resource table, String itemCode) throws DatabaseException { Object graph = SCLContext.getCurrent().get("graph"); if (graph != null && graph instanceof ReadGraph) return getTableItem((ReadGraph) graph, table, itemCode); else return getTableItem(Simantics.getSession(), table, itemCode); } /** * Reset all components that address the given tech type table to the table values. * * @param table A tech type table * @throws DatabaseException */ public static void resetComponents(Resource table) throws DatabaseException { Simantics.getSession().syncRequest(new WriteRequest() { @Override public void perform(WriteGraph graph) throws DatabaseException { Resource model = graph.syncRequest(new PossibleIndexRoot(table), TransientCacheListener.instance()); if (model == null) return; Resource type = graph.getPossibleObject(table, DistrictNetworkResource.getInstance(graph).TechType_TechTypeTable_HasComponentType); if (type == null) return; Collection components = QueryIndexUtils.searchByType(graph, model, type); for (Resource component : components) { updateComponent(graph, component, table); } } }); } /** * Reset all diagram elements to tech type table values. * * @param table A tech type table * @throws DatabaseException */ public static void resetMapElements(Resource table) throws DatabaseException { Simantics.getSession().syncRequest(new WriteRequest() { @Override public void perform(WriteGraph graph) throws DatabaseException { Resource model = graph.syncRequest(new PossibleIndexRoot(table), TransientCacheListener.instance()); if (model == null) return; DistrictNetworkResource DN = DistrictNetworkResource.getInstance(graph); Resource type = graph.getPossibleObject(table, DN.TechType_TechTypeTable_HasComponentType); if (type == null) return; Layer0 L0 = Layer0.getInstance(graph); String typeName = graph.getRelatedValue2(type, L0.HasName); Resource mapping = null; for (Resource m : QueryIndexUtils.searchByType(graph, model, DN.Mapping_Base)) { String name = graph.getRelatedValue2(m, DN.Mapping_ComponentType); if (Objects.equals(name, typeName)) { mapping = m; break; } } if (mapping == null) { LOGGER.warn("No mapping found for component type {}", type); return; } Map properties = new HashMap<>(); Resource mappingType = graph.getSingleType(mapping, DN.Mapping_Base); Collection mappingRelations = graph.getObjects(mappingType, L0.DomainOf); for (Resource r : mappingRelations) { String propertyName = graph.getPossibleRelatedValue2(mapping, r); if (propertyName == null) continue; Resource relation = graph.getPossibleObject(r, DN.Mapping_HasPropertyRelation); if (relation == null) continue; properties.put(propertyName, graph.syncRequest(new PropertyInfoRequest(relation))); } Map> data = graph.syncRequest(new TechTypeTableData(table), TransientCacheListener.instance()); String keyName = graph.syncRequest(new TechTypeTableKeyName(table), TransientCacheListener.instance()); Resource keyRelation = properties.get(keyName).predicate; if (keyRelation == null) { LOGGER.warn("No relation mapped to property {} found in {} mapping", keyName, mapping); return; } Resource elementType = graph.isInstanceOf(mapping, DN.Mapping_EdgeMapping) ? DN.Edge : graph.isInstanceOf(mapping, DN.Mapping_VertexMapping) ? DN.Vertex : DN.Element; Collection elements = QueryIndexUtils.searchByType(graph, model, elementType); for (Resource element : elements) { Resource elementMapping = graph.getPossibleObject(element, DN.HasMapping); if (!mapping.equals(elementMapping)) continue; String key = graph.getPossibleRelatedValue2(element, keyRelation); Map values = data.get(key); if (values == null) { LOGGER.info("Key {} no found in tech type table {}", key, table); continue; } for (Map.Entry entry : values.entrySet()) { PropertyInfo prop = properties.get(entry.getKey()); if (prop == null) continue; String value = entry.getValue(); Datatype dt = prop.requiredDatatype; if (dt instanceof NumberType) { try { Object num = ((NumberBinding)prop.defaultBinding).create(value.replace(",", ".")); graph.claimLiteral(element, prop.predicate, prop.literalRange, num, prop.defaultBinding); } catch (NumberFormatException e) { // Revert to asserted value graph.deny(element, prop.predicate); } catch (BindingException e) { LOGGER.error("Failed to get binding for datatype {}", dt, e); } } else if (dt instanceof StringType) { graph.claimLiteral(element, prop.predicate, prop.literalRange, value, Bindings.STRING); } else { LOGGER.warn("updateComponent: Unsupported property type {}", dt); } } } } }); } /** * Update property values of a component based on the values in an associated tech type table, if any * * @param graph A write access interface to the graph database * @param component A structural component * @throws DatabaseException */ public static void updateComponent(WriteGraph graph, Resource component) throws DatabaseException { StructuralResource2 STR = StructuralResource2.getInstance(graph); Resource type = graph.getSingleType(component, STR.Component); Resource model = graph.syncRequest(new PossibleIndexRoot(component)); if (model == null) { LOGGER.info("updateComponent: No model for {}", component); return; } Resource table = graph.syncRequest(new PossibleTechTypeTable(model, type), TransientCacheListener.instance()); if (table == null) { LOGGER.info("updateComponent: No tech type table for {} in {}", type, model); return; } updateComponent(graph, component, table); } /** * Update property values of a component based on the values in an associated tech type table, if any * * @param component A structural component * @throws DatabaseException */ public static void updateComponent(Resource component) throws DatabaseException { if (LOGGER.isTraceEnabled()) LOGGER.trace("updateComponent({})", component); Simantics.getSession().syncRequest(new WriteRequest() { @Override public void perform(WriteGraph graph) throws DatabaseException { updateComponent(graph, component); } }); } /** * Update a single component's property values to those selected from a tech * type table * * @param graph A write interface to the db * @param component The component to be updated * @param table A tech type table * @throws DatabaseException */ public static void updateComponent(WriteGraph graph, Resource component, Resource table) throws DatabaseException { if (LOGGER.isTraceEnabled()) LOGGER.trace("updateComponent(graph, {}, {}), component, table)"); Variable v = graph.syncRequest(new PossibleVariable(component)); if (v == null) { LOGGER.info("No variable found for {}", component); return; } String keyProp = getKeyPropertyName(graph, table); String itemCode = v.getPropertyValue(graph, keyProp); Map map = graph.syncRequest(new PossibleTechTypeItem(table, itemCode), TransientCacheListener.instance()); if (map == null) { LOGGER.info("No entry found for \"{}\" in tech type table {}", itemCode, table); return; } for (String key : map.keySet()) { // Don't write the key property if (key.equals(DEFAULT_KEY_NAME)) continue; Variable prop = v.getPossibleProperty(graph, key); if (prop == null) continue; Datatype dt = prop.getPossibleDatatype(graph); if (dt == null) continue; String value = map.get(key); if (dt instanceof NumberType) { try { Double num = Double.valueOf(value.replace(",", ".")); NumberBinding binding = Bindings.getBinding(dt); prop.setValue(graph, binding.create(num)); } catch (NumberFormatException e) { // Revert to asserted value Resource pred = prop.getPossiblePredicateResource(graph); if (pred != null) graph.deny(component, pred); } catch (BindingException e) { LOGGER.error("Failed to get binding for datatype {}", dt, e); } } else if (dt instanceof StringType) { prop.setValue(graph, value); } else { LOGGER.warn("updateComponent: Unsupported property type {}", dt); } } } public static String getKeyPropertyName(ReadGraph graph, Resource table) throws DatabaseException { String name = graph.syncRequest(new TechTypeTableKeyName(table), TransientCacheListener.instance()); if (name == null) name = "_" + DEFAULT_KEY_NAME; return name; } }