package org.molgenis.js.magma; import com.google.api.client.util.Maps; import com.google.common.base.Stopwatch; import org.molgenis.data.Entity; import org.molgenis.data.meta.AttributeType; import org.molgenis.data.meta.model.Attribute; import org.molgenis.file.model.FileMeta; import org.molgenis.js.nashorn.NashornScriptEngine; import org.molgenis.script.ScriptException; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; import java.sql.Date; import java.sql.Timestamp; import java.util.Map; import static java.lang.String.format; import static java.util.Objects.requireNonNull; import static java.util.concurrent.TimeUnit.MICROSECONDS; import static java.util.stream.Collectors.toList; import static java.util.stream.StreamSupport.stream; /** * JavaScript script evaluator using the Nashorn script engine. */ @Component public class JsMagmaScriptEvaluator { private static final Logger LOG = LoggerFactory.getLogger(JsMagmaScriptEvaluator.class); private final NashornScriptEngine jsScriptEngine; @Autowired public JsMagmaScriptEvaluator(NashornScriptEngine jsScriptEngine) { this.jsScriptEngine = requireNonNull(jsScriptEngine); } /** * Evaluate a expression for the given entity. * * @param expression JavaScript expression * @param entity entity * @return evaluated expression result, return type depends on the expression. */ public Object eval(String expression, Entity entity) { Stopwatch stopwatch = null; if (LOG.isTraceEnabled()) { stopwatch = Stopwatch.createStarted(); } Map<String, Object> scriptEngineValueMap = toScriptEngineValueMap(entity); Object value; try { value = jsScriptEngine.invokeFunction("evalScript", expression, scriptEngineValueMap); } catch (Throwable t) { return new ScriptException(t); } if (stopwatch != null) { stopwatch.stop(); LOG.trace("Script evaluation took {} µs", stopwatch.elapsed(MICROSECONDS)); } return value; } private static Map<String, Object> toScriptEngineValueMap(Entity entity) { Map<String, Object> map = Maps.newHashMap(); entity.getEntityType().getAtomicAttributes() .forEach(attr -> map.put(attr.getName(), toScriptEngineValue(entity, attr))); return map; } private static Object toScriptEngineValue(Entity entity, Attribute attr) { Object value; String attrName = attr.getName(); AttributeType attrType = attr.getDataType(); switch (attrType) { case BOOL: value = entity.getBoolean(attrName); break; case CATEGORICAL: case XREF: Entity xrefEntity = entity.getEntity(attrName); value = xrefEntity != null ? toScriptEngineValue(xrefEntity, xrefEntity.getEntityType().getIdAttribute()) : null; break; case CATEGORICAL_MREF: case MREF: case ONE_TO_MANY: Iterable<Entity> mrefEntities = entity.getEntities(attrName); value = stream(mrefEntities.spliterator(), false) .map(mrefEntity -> toScriptEngineValue(mrefEntity, mrefEntity.getEntityType().getIdAttribute())) .collect(toList()); break; case DATE: // convert to epoch Date date = entity.getDate(attrName); value = date != null ? date.getTime() : null; break; case DATE_TIME: // convert to epoch Timestamp timestamp = entity.getTimestamp(attrName); value = timestamp != null ? timestamp.getTime() : null; break; case DECIMAL: value = entity.getDouble(attrName); break; case EMAIL: case ENUM: case HTML: case HYPERLINK: case SCRIPT: case STRING: case TEXT: value = entity.getString(attrName); break; case FILE: FileMeta fileEntity = entity.getEntity(attrName, FileMeta.class); value = fileEntity != null ? toScriptEngineValue(fileEntity, fileEntity.getEntityType().getIdAttribute()) : null; break; case INT: value = entity.getInt(attrName); break; case LONG: value = entity.getLong(attrName); break; case COMPOUND: throw new RuntimeException(format("Illegal attribute type [%s]", attrType.toString())); default: throw new RuntimeException(format("Unknown attribute type [%s]", attrType.toString())); } return value; } }