/** * Copyright (C) 2012 - present by OpenGamma Inc. and the OpenGamma group of companies * * Please see distribution for license. */ package com.opengamma.financial.analytics; import java.util.Collections; import java.util.Map; import java.util.Set; import org.joda.beans.Bean; import org.joda.beans.MetaBean; import org.joda.beans.MetaProperty; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import com.google.common.collect.Iterables; import com.opengamma.OpenGammaRuntimeException; import com.opengamma.core.security.Security; import com.opengamma.engine.ComputationTarget; import com.opengamma.engine.function.AbstractFunction; import com.opengamma.engine.function.FunctionCompilationContext; import com.opengamma.engine.function.FunctionExecutionContext; import com.opengamma.engine.function.FunctionInputs; import com.opengamma.engine.target.ComputationTargetType; import com.opengamma.engine.value.ComputedValue; import com.opengamma.engine.value.ValueProperties; import com.opengamma.engine.value.ValueRequirement; import com.opengamma.engine.value.ValueRequirementNames; import com.opengamma.engine.value.ValueSpecification; import com.opengamma.financial.OpenGammaCompilationContext; /** * */ public class AttributesFunction extends AbstractFunction.NonCompiledInvoker { /** The attribute property name */ public static final String PROPERTY_ATTRIBUTE_NAME = "AttributeName"; private static final Logger s_logger = LoggerFactory.getLogger(AttributesFunction.class); @Override public Set<ComputedValue> execute(final FunctionExecutionContext executionContext, final FunctionInputs inputs, final ComputationTarget target, final Set<ValueRequirement> desiredValues) { final ValueRequirement desiredValue = Iterables.getOnlyElement(desiredValues); final ValueProperties properties = desiredValue.getConstraints(); final String attributeName = properties.getSingleValue(PROPERTY_ATTRIBUTE_NAME); final Security security = target.getSecurity(); final Map<String, String> attributes = security.getAttributes(); String result = attributes.get(attributeName); if (result == null) { if (security instanceof Bean) { Bean ms = (Bean) security; MetaBean metaBean = ms.metaBean(); if (metaBean.metaPropertyExists(attributeName)) { result = metaBean.metaProperty(attributeName).getString(ms); } } } if (result == null) { throw new OpenGammaRuntimeException("Could not get value for attribute " + attributeName); } final ValueSpecification spec = new ValueSpecification(ValueRequirementNames.ATTRIBUTES, target.toSpecification(), properties); return Collections.singleton(new ComputedValue(spec, result)); } @Override public ComputationTargetType getTargetType() { return ComputationTargetType.SECURITY; } @Override public Set<ValueSpecification> getResults(final FunctionCompilationContext context, final ComputationTarget target) { final Security security = target.getSecurity(); final Map<String, String> attributes = security.getAttributes(); if (attributes.isEmpty() && !(security instanceof Bean)) { // No explicit attributes, and can't query a non-Bean security return null; } final ValueProperties.Builder properties = createValueProperties(); if (!attributes.isEmpty()) { for (String attribute : attributes.keySet()) { properties.with(PROPERTY_ATTRIBUTE_NAME, attribute); } } if (security instanceof Bean) { Bean bean = (Bean) security; MetaBean metaBean = bean.metaBean(); for (MetaProperty<?> property : metaBean.metaPropertyIterable()) { properties.with(PROPERTY_ATTRIBUTE_NAME, property.name()); } } return Collections.singleton(new ValueSpecification(ValueRequirementNames.ATTRIBUTES, target.toSpecification(), properties.get())); } @Override public Set<ValueRequirement> getRequirements(final FunctionCompilationContext context, final ComputationTarget target, final ValueRequirement desiredValue) { final ValueProperties constraints = desiredValue.getConstraints(); final Set<String> attributeNames = constraints.getValues(PROPERTY_ATTRIBUTE_NAME); if (!OpenGammaCompilationContext.isPermissive(context)) { if (attributeNames == null || attributeNames.size() != 1) { s_logger.error("Did not specify a single attribute name"); return null; } } return Collections.emptySet(); } }