package se.cambio.cds.util; import org.apache.commons.jexl2.Expression; import org.apache.commons.jexl2.JexlContext; import org.apache.commons.jexl2.JexlEngine; import org.apache.commons.jexl2.MapContext; import org.apache.log4j.Logger; import org.openehr.rm.datatypes.basic.DataValue; import org.openehr.rm.datatypes.quantity.datetime.DvDateTime; import org.openehr.rm.datatypes.text.CodePhrase; import org.openehr.rm.datatypes.text.DvCodedText; import se.cambio.cds.gdl.model.Binding; import se.cambio.cds.gdl.model.Guide; import se.cambio.cds.gdl.model.TermBinding; import se.cambio.cds.gdl.model.expression.ExpressionItem; import se.cambio.cds.gdl.model.expression.OperatorKind; import se.cambio.cds.model.facade.execution.vo.GeneratedArchetypeReference; import se.cambio.cds.model.facade.execution.vo.GeneratedElementInstance; import se.cambio.cds.model.facade.execution.vo.PredicateGeneratedElementInstance; import se.cambio.cds.model.facade.execution.vo.RuleReference; import se.cambio.cds.model.instance.ArchetypeReference; import se.cambio.cds.model.instance.ElementInstance; import se.cambio.openehr.controller.session.OpenEHRSessionManager; import se.cambio.openehr.util.ExceptionHandler; import se.cambio.openehr.util.OpenEHRConst; import se.cambio.openehr.util.exceptions.InternalErrorException; import se.cambio.openehr.util.misc.DataValueGenerator; import java.util.*; public class ElementInstanceCollectionUtil { public static boolean isEmpty(ArchetypeReference ar){ for (String idElement : ar.getElementInstancesMap().keySet()) { ElementInstance ei = ar.getElementInstancesMap().get(idElement); if (ei.getDataValue()!=null){ return false; } } return true; } public static ArchetypeReference getEmptyArchetypeReference(Set<ArchetypeReference> archetypeReferences){ for (ArchetypeReference archetypeReference : archetypeReferences) { if (isEmpty(archetypeReference)){ return archetypeReference; } } return null; } public static boolean containsAll(ArchetypeReference ar1, ArchetypeReference ar2){ return ar1.getElementInstancesMap().keySet().containsAll(ar2.getElementInstancesMap().keySet()); } /** * Looks if the second reference matches the first one, if empty references are found, they will be copied into the second one (only if the rest is matched). * @param ar1 * @param ar2 * @param guideMap * @param date * @return true if ar1 matches ar2 */ public static boolean matchAndFill(GeneratedArchetypeReference ar1, ArchetypeReference ar2, Map<String, Guide> guideMap, Calendar date){ Collection<ElementInstance> emptyElementInstances = new ArrayList<ElementInstance>(); boolean matches = matches(ar1, ar2, guideMap, date); if (!matches){ return false; }//else continue with the filling //Set AR to empty elementInstances found for (String idElement : ar1.getElementInstancesMap().keySet()) { ElementInstance ei1 = ar1.getElementInstancesMap().get(idElement); ElementInstance ei2 = ar2.getElementInstancesMap().get(idElement); if (!(ei1 instanceof PredicateGeneratedElementInstance) && ei2==null){ ei2 = ei1.clone(); emptyElementInstances.add(ei2); } if (ei1 instanceof GeneratedElementInstance && ei2 instanceof GeneratedElementInstance){ ((GeneratedElementInstance)ei2).getRuleReferences().addAll(((GeneratedElementInstance) ei1).getRuleReferences()); } } for (ElementInstance elementInstance : emptyElementInstances) { elementInstance.setArchetypeReference(ar2); } return true; } public static boolean matches(GeneratedArchetypeReference ar1, ArchetypeReference ar2, Map<String, Guide> guideMap, Calendar date){ if (!ar1.getIdArchetype().equals(ar2.getIdArchetype())){ return false; }else{ for (String idElement : ar1.getElementInstancesMap().keySet()) { ElementInstance ei1 = ar1.getElementInstancesMap().get(idElement); ElementInstance ei2 = ar2.getElementInstancesMap().get(idElement); if (ei1 instanceof PredicateGeneratedElementInstance){ if (ei2!=null){ PredicateGeneratedElementInstance pgei = ((PredicateGeneratedElementInstance)ei1); Set<Guide> guides = new HashSet<Guide>(); for(RuleReference ruleReference: pgei.getRuleReferences()){ Guide guide = guideMap.get(ruleReference.getGuideId()); if (guide==null){ Logger.getLogger(ElementInstanceCollectionUtil.class).warn("Null guideline for rule reference '"+ruleReference+"'"); }else{ guides.add(guide); } } DataValue dv = null; if (pgei.getOperatorKind().equals(OperatorKind.IS_A)){ dv = ei1.getDataValue(); //We do not resolve here IS_A codes, we do that during the dv matching } else { dv = resolvePredicate(ei1.getDataValue(), pgei.getOperatorKind(), guides, date); } if (!matches(dv, ei2.getDataValue(), pgei.getOperatorKind(), guides)){ return false; } }else{ return false; } } } return true; } } public static boolean matches( DataValue dv1, DataValue dv2, OperatorKind operatorKind, Collection<Guide> guides){ if (OperatorKind.IS_A.equals(operatorKind)){ if (dv1 instanceof DvCodedText && dv2 instanceof DvCodedText){ CodePhrase elementCodePhrase = ((DvCodedText)dv2).getDefiningCode(); CodePhrase predicateCodePhrase = ((DvCodedText)dv1).getDefiningCode(); Set<CodePhrase> codePhrases = new HashSet<CodePhrase>(); if (guides!=null){ for(Guide guide: guides){ if (guide.getOntology().getTermBindings()!=null){ for (String terminologyId : guide.getOntology().getTermBindings().keySet()) { TermBinding termBinding = guide.getOntology().getTermBindings().get(terminologyId); if (termBinding!=null){ Binding binding = termBinding.getBindings().get(predicateCodePhrase.getCodeString()); if (binding!=null && binding.getCodes()!=null){ codePhrases.addAll(binding.getCodes()); } } } } } }else{ codePhrases.add(predicateCodePhrase); } if (!codePhrases.isEmpty()){ try{ return OpenEHRSessionManager.getTerminologyFacadeDelegate().isSubclassOf(elementCodePhrase, codePhrases); }catch(Exception e){ Logger.getLogger(ElementInstanceCollectionUtil.class).warn(e.getMessage()); return false; } }else{ return false; } } }else if (OperatorKind.EQUALITY.equals(operatorKind)){ return DVUtil.equalDVs(dv1, dv2); }else if (OperatorKind.INEQUAL.equals(operatorKind)){ return !DVUtil.equalDVs(dv1, dv2); }else if (OperatorKind.MAX.equals(operatorKind)|| (OperatorKind.MIN.equals(operatorKind))){ if(dv2==null){ return false; }else{ return true; } }else if (OperatorKind.GREATER_THAN.equals(operatorKind)){ if(dv2==null){ return false; }else{ return DVUtil.compareDVs(dv1, dv2)<0; } }else if (OperatorKind.GREATER_THAN_OR_EQUAL.equals(operatorKind)){ if(dv2==null){ return false; }else{ return DVUtil.compareDVs(dv1, dv2)<=0; } }else if (OperatorKind.LESS_THAN.equals(operatorKind)){ if(dv2==null){ return false; }else{ return DVUtil.compareDVs(dv1, dv2)>0; } }else if (OperatorKind.LESS_THAN_OR_EQUAL.equals(operatorKind)){ if(dv2==null){ return false; }else{ return DVUtil.compareDVs(dv1, dv2)>=0; } } return false; } public static DataValue resolvePredicate(DataValue dv, OperatorKind op, Collection<Guide> guides, Calendar date){ if (OperatorKind.IS_A.equals(op)){ if (dv instanceof DvCodedText){ DvCodedText dvCT = (DvCodedText)dv; if (guides!=null){ for(Guide guide: guides){ if (guide!=null && guide.getOntology().getTermBindings()!=null){ for (String terminologyId : guide.getOntology().getTermBindings().keySet()) { TermBinding termBinding = guide.getOntology().getTermBindings().get(terminologyId); if (termBinding!=null){ Binding binding = termBinding.getBindings().get(dvCT.getDefiningCode().getCodeString()); if (binding!=null && binding.getCodes()!=null && !binding.getCodes().isEmpty()){ CodePhrase cf = binding.getCodes().get(0); return new DvCodedText(dvCT.getValue(),cf); } } } } } } //If reaches here, no terminology was found (problem) Logger.getLogger(ElementInstanceCollectionUtil.class).warn("No terminology binding for '"+dv+"' was found! (num guides="+(guides==null?"0":guides.size())+")"); return null; }else{ Logger.getLogger(ElementInstanceCollectionUtil.class).warn("Not a coded text '"+dv+"'"); return null; } }else if (dv instanceof CurrentTimeExpressionDataValue){ CurrentTimeExpressionDataValue ctedv = ((CurrentTimeExpressionDataValue)dv); ExpressionItem expressionItem = ctedv.getExpressionItem(); String attribute = ctedv.getAttrbute(); try { String expStr = ExpressionUtil.getArithmeticExpressionStr(null, expressionItem, null); date = (date!=null?date:Calendar.getInstance()); DvDateTime currentDateTime = DataValueGenerator.toDvDateTime(date); JexlEngine engine = new JexlEngine(); Expression e = engine.createExpression(expStr); JexlContext context = new MapContext(); context.set("$"+OpenEHRConst.CURRENT_DATE_TIME_ID, currentDateTime); Object obj = e.evaluate(context); if (obj instanceof Double){ obj = ((Double)obj).longValue(); //In dates we never need double value } currentDateTime = (DvDateTime)DataValueGenerator.createDV(currentDateTime, attribute, obj); return currentDateTime; } catch (InternalErrorException e) { ExceptionHandler.handle(e); } return dv; }else{ return dv; } } } /* * ***** BEGIN LICENSE BLOCK ***** * Version: MPL 2.0/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Mozilla Public 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.mozilla.org/MPL/ * * Software distributed under the License is distributed on an 'AS IS' basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License * for the specific language governing rights and limitations under the * License. * * * The Initial Developers of the Original Code are Iago Corbal and Rong Chen. * Portions created by the Initial Developer are Copyright (C) 2012-2013 * the Initial Developer. All Rights Reserved. * * Contributor(s): * * Software distributed under the License is distributed on an 'AS IS' basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License * for the specific language governing rights and limitations under the * License. * * ***** END LICENSE BLOCK ***** */