/** * Copyright (C) 2012 - present by OpenGamma Inc. and the OpenGamma group of companies * * Please see distribution for license. */ package com.opengamma.financial.value; import java.util.Map; import java.util.Set; import com.google.common.collect.ImmutableSet; import com.google.common.collect.Iterables; 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.ValueProperties.Builder; import com.opengamma.engine.value.ValueRequirement; import com.opengamma.engine.value.ValueSpecification; import com.opengamma.util.ArgumentChecker; import com.opengamma.util.async.AsynchronousExecution; /** * A generic function for exposing a particular value requirement under a different name. This may allow the value to be delivered through a heavily constrained value requirement, but output through a * simple value requirement name. */ public class ValueRequirementAliasFunction extends AbstractFunction.NonCompiledInvoker { // REVIEW 2013-04-12 Andrew - This is a poor implementation that doesn't propogate constraints particularly satisfactorily; use ValueRenamingFunction instead private final String _aliasedValueRequirementName; private final String _inputRequirementName; private final ValueProperties _inputConstraints; private final Set<String> _preservedProperties; private final ComputationTargetType _targetType; /** * Constructs an instance. * * @param aliasedValueRequirementName the value requirement name under which to expose the input value, not null * @param inputRequirementName the input value requirement name, not null * @param inputConstraints the static input value requirement constraints, not null * @param preservedProperties the properties preserved on the output which may be constrained, not null * @param targetType the function target type, not null */ public ValueRequirementAliasFunction(String aliasedValueRequirementName, String inputRequirementName, ValueProperties inputConstraints, Set<String> preservedProperties, ComputationTargetType targetType) { ArgumentChecker.notNull(aliasedValueRequirementName, "aliasedValueRequirementName"); ArgumentChecker.notNull(inputRequirementName, "inputRequirementName"); ArgumentChecker.notNull(inputConstraints, "inputConstraints"); ArgumentChecker.notNull(preservedProperties, "preservedProperties"); ArgumentChecker.notNull(targetType, "targetType"); _aliasedValueRequirementName = aliasedValueRequirementName; _inputRequirementName = inputRequirementName; _inputConstraints = inputConstraints; _preservedProperties = preservedProperties; _targetType = targetType; } @Override public ComputationTargetType getTargetType() { return _targetType; } @Override public Set<ValueSpecification> getResults(FunctionCompilationContext context, ComputationTarget target) { final Builder builder = createValueProperties(); for (String preservedProperty : _preservedProperties) { builder.withAny(preservedProperty); } return ImmutableSet.of(new ValueSpecification(_aliasedValueRequirementName, target.toSpecification(), builder.get())); } @Override public Set<ValueRequirement> getRequirements(FunctionCompilationContext context, ComputationTarget target, ValueRequirement desiredValue) { return ImmutableSet.of(createValueRequirement(target, desiredValue)); } @Override public Set<ValueSpecification> getResults(FunctionCompilationContext context, ComputationTarget target, Map<ValueSpecification, ValueRequirement> inputs) { final ValueSpecification inputSpec = Iterables.getOnlyElement(inputs.keySet()); final Builder builder = createValueProperties(); appendPreservedProperties(inputSpec.getProperties(), builder); return ImmutableSet.of(new ValueSpecification(_aliasedValueRequirementName, target.toSpecification(), builder.get())); } @Override public Set<ComputedValue> execute(FunctionExecutionContext executionContext, FunctionInputs inputs, ComputationTarget target, Set<ValueRequirement> desiredValues) throws AsynchronousExecution { final Object result = inputs.getValue(_inputRequirementName); final ValueRequirement desiredValue = desiredValues.iterator().next(); return ImmutableSet.of(new ComputedValue(new ValueSpecification(desiredValue.getValueName(), target.toSpecification(), desiredValue.getConstraints()), result)); } private ValueRequirement createValueRequirement(ComputationTarget target, ValueRequirement desiredValue) { final Builder constraints = _inputConstraints.copy(); appendPreservedProperties(desiredValue.getConstraints(), constraints); return new ValueRequirement(_inputRequirementName, target.toSpecification(), constraints.get()); } private void appendPreservedProperties(ValueProperties from, final Builder to) { Set<String> sourceProperties = from.getProperties(); if (sourceProperties == null) { return; } for (String constraintName : sourceProperties) { if (!_preservedProperties.contains(constraintName)) { continue; } Set<String> constraintValues = from.getValues(constraintName); if (constraintValues.isEmpty()) { to.withAny(constraintName); } else { to.with(constraintName, constraintValues); } } } }