/* * 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. * * Contributions from 2013-2017 where performed either by US government * employees, or under US Veterans Health Administration contracts. * * US Veterans Health Administration contributions by government employees * are work of the U.S. Government and are not subject to copyright * protection in the United States. Portions contributed by government * employees are USGovWork (17USC ยง105). Not subject to copyright. * * Contribution by contractors to the US Veterans Health Administration * during this period are contractually contributed under the * Apache License, Version 2.0. * * See: https://www.usa.gov/government-works * * Contributions prior to 2013: * * Copyright (C) International Health Terminology Standards Development Organisation. * Licensed under the Apache License, Version 2.0. * */ package sh.isaac.model.logic.node; //~--- JDK imports ------------------------------------------------------------ import java.io.ByteArrayOutputStream; import java.io.DataInputStream; import java.io.DataOutput; import java.io.DataOutputStream; import java.io.IOException; import java.util.ArrayList; import java.util.List; import java.util.SortedSet; import java.util.TreeSet; import java.util.UUID; //~--- non-JDK imports -------------------------------------------------------- import sh.isaac.api.DataTarget; import sh.isaac.api.collections.ConceptSequenceSet; import sh.isaac.api.logic.LogicNode; import sh.isaac.api.tree.TreeNodeVisitData; import sh.isaac.model.logic.LogicalExpressionOchreImpl; //~--- classes ---------------------------------------------------------------- /** * Created by kec on 12/10/14. */ public abstract class AbstractLogicNode implements LogicNode { /** The Constant namespaceUuid. */ protected static final UUID namespaceUuid = UUID.fromString("d64c6d91-a37d-11e4-bcd8-0800200c9a66"); //~--- fields -------------------------------------------------------------- /** The node index. */ private short nodeIndex = Short.MIN_VALUE; /** The node uuid. */ protected UUID nodeUuid = null; /** The logic graph version. */ LogicalExpressionOchreImpl logicGraphVersion; //~--- constructors -------------------------------------------------------- /** * Instantiates a new abstract logic node. * * @param anotherNode the another node */ protected AbstractLogicNode(AbstractLogicNode anotherNode) { this.nodeIndex = anotherNode.nodeIndex; this.nodeUuid = anotherNode.nodeUuid; } /** * Instantiates a new abstract logic node. * * @param logicGraphVersion the logic graph version */ public AbstractLogicNode(LogicalExpressionOchreImpl logicGraphVersion) { this.logicGraphVersion = logicGraphVersion; logicGraphVersion.addNode(this); } /** * Instantiates a new abstract logic node. * * @param logicGraphVersion the logic graph version * @param dataInputStream the data input stream * @throws IOException Signals that an I/O exception has occurred. */ public AbstractLogicNode(LogicalExpressionOchreImpl logicGraphVersion, DataInputStream dataInputStream) throws IOException { this.nodeIndex = dataInputStream.readShort(); this.logicGraphVersion = logicGraphVersion; logicGraphVersion.addNode(this); } //~--- methods ------------------------------------------------------------- /** * Should be overridden by subclasses that need to add concepts. * Concepts from connector nodes should not be added. * * @param conceptSequenceSet the concept sequence set */ @Override public void addConceptsReferencedByNode(ConceptSequenceSet conceptSequenceSet) { conceptSequenceSet.add(getNodeSemantic().getConceptSequence()); } /** * Compare to. * * @param o the o * @return the int */ @Override public int compareTo(LogicNode o) { if (this.getNodeSemantic() != o.getNodeSemantic()) { return this.getNodeSemantic() .compareTo(o.getNodeSemantic()); } return compareFields(o); } /** * Equals. * * @param o the o * @return true, if successful */ @Override public boolean equals(Object o) { if (this == o) { return true; } return !((o == null) || (getClass() != o.getClass())); } /** * Fragment to string. * * @return A string representing the fragment of the expression * rooted in this node. */ @Override public String fragmentToString() { return fragmentToString(""); } /** * Fragment to string. * * @param nodeIdSuffix the node id suffix * @return the string */ @Override public String fragmentToString(String nodeIdSuffix) { final StringBuilder builder = new StringBuilder(); this.logicGraphVersion.processDepthFirst(this, (LogicNode logicNode, TreeNodeVisitData graphVisitData) -> { for (int i = 0; i < graphVisitData.getDistance(logicNode.getNodeIndex()); i++) { builder.append(" "); } builder.append(logicNode.toString(nodeIdSuffix)); builder.append("\n"); }); return builder.toString(); } /** * Hash code. * * @return the int */ @Override public int hashCode() { return this.nodeIndex; } /** * Sort. */ @Override public void sort() { // override on nodes with multiple children. } /** * To string. * * @return the string */ @Override public String toString() { return toString(""); } /** * To string. * * @param nodeIdSuffix the node id suffix * @return the string */ @Override public String toString(String nodeIdSuffix) { return ""; } /** * Compare fields. * * @param o the o * @return the int */ protected abstract int compareFields(LogicNode o); /** * Inits the node uuid. * * @return the uuid */ protected abstract UUID initNodeUuid(); /** * Write data. * * @param dataOutput the data output * @param dataTarget the data target * @throws IOException Signals that an I/O exception has occurred. */ protected void writeData(DataOutput dataOutput, DataTarget dataTarget) throws IOException {} /** * Write node data. * * @param dataOutput the data output * @param dataTarget the data target * @throws IOException Signals that an I/O exception has occurred. */ protected abstract void writeNodeData(DataOutput dataOutput, DataTarget dataTarget) throws IOException; //~--- get methods --------------------------------------------------------- /** * Gets the bytes. * * @param dataTarget the data target * @return the bytes */ @Override public byte[] getBytes(DataTarget dataTarget) { try { try (ByteArrayOutputStream outputStream = new ByteArrayOutputStream()) { final DataOutputStream output = new DataOutputStream(outputStream); output.writeByte(getNodeSemantic().ordinal()); output.writeShort(this.nodeIndex); writeNodeData(output, dataTarget); return outputStream.toByteArray(); } } catch (final IOException e) { throw new RuntimeException(e); } } /** * Gets the children. * * @return the children */ @Override public abstract AbstractLogicNode[] getChildren(); /** * Gets the descendents. * * @return the descendents */ @Override public AbstractLogicNode[] getDescendents() { final List<AbstractLogicNode> descendents = new ArrayList<>(); getDescendents(this, descendents); return descendents.toArray(new AbstractLogicNode[descendents.size()]); } /** * Gets the descendents. * * @param parent the parent * @param descendents the descendents * @return the descendents */ private void getDescendents(AbstractLogicNode parent, List<AbstractLogicNode> descendents) { for (final AbstractLogicNode child: parent.getChildren()) { descendents.add(child); getDescendents(child, descendents); } } /** * Gets the node index. * * @return the node index */ @Override public short getNodeIndex() { return this.nodeIndex; } //~--- set methods --------------------------------------------------------- /** * Sets the node index. * * @param nodeIndex the new node index */ @Override public void setNodeIndex(short nodeIndex) { if (this.nodeIndex == Short.MIN_VALUE) { this.nodeIndex = nodeIndex; } else if (this.nodeIndex == nodeIndex) { // nothing to do... } else { throw new IllegalStateException("LogicNode index cannot be changed once set. NodeId: " + this.nodeIndex + " attempted: " + nodeIndex); } } //~--- get methods --------------------------------------------------------- /** * Gets the node uuid. * * @return the node uuid */ protected UUID getNodeUuid() { if (this.nodeUuid == null) { this.nodeUuid = initNodeUuid(); } return this.nodeUuid; } ; //~--- get methods --------------------------------------------------------- /** * Gets the node uuid set for depth. * * @param depth the depth * @return the node uuid set for depth */ public SortedSet<UUID> getNodeUuidSetForDepth(int depth) { final SortedSet<UUID> uuidSet = new TreeSet<>(); uuidSet.add(getNodeUuid()); if (depth > 1) { for (final AbstractLogicNode child: getChildren()) { uuidSet.addAll(child.getNodeUuidSetForDepth(depth - 1)); } } return uuidSet; } }