All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.opengamma.strata.market.param.UnitParameterSensitivity Maven / Gradle / Ivy

/*
 * Copyright (C) 2015 - present by OpenGamma Inc. and the OpenGamma group of companies
 *
 * Please see distribution for license.
 */
package com.opengamma.strata.market.param;

import java.io.Serializable;
import java.util.List;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Optional;
import java.util.function.DoubleUnaryOperator;
import java.util.stream.Stream;

import org.joda.beans.Bean;
import org.joda.beans.BeanBuilder;
import org.joda.beans.ImmutableBean;
import org.joda.beans.JodaBeanUtils;
import org.joda.beans.MetaBean;
import org.joda.beans.MetaProperty;
import org.joda.beans.gen.BeanDefinition;
import org.joda.beans.gen.ImmutableValidator;
import org.joda.beans.gen.PropertyDefinition;
import org.joda.beans.impl.direct.DirectMetaBean;
import org.joda.beans.impl.direct.DirectMetaProperty;
import org.joda.beans.impl.direct.DirectMetaPropertyMap;
import org.joda.beans.impl.direct.DirectPrivateBeanBuilder;

import com.google.common.collect.ComparisonChain;
import com.google.common.collect.ImmutableList;
import com.opengamma.strata.basics.currency.Currency;
import com.opengamma.strata.collect.ArgChecker;
import com.opengamma.strata.collect.Messages;
import com.opengamma.strata.collect.array.DoubleArray;
import com.opengamma.strata.data.MarketDataName;
import com.opengamma.strata.market.curve.Curve;
import com.opengamma.strata.market.surface.Surface;

/**
 * Unit parameter sensitivity for parameterized market data, such as a curve.
 * 

* Parameter sensitivity is the sensitivity of a value to the parameters of a * {@linkplain ParameterizedData parameterized market data} object that is used to determine the value. * Common {@code ParameterizedData} implementations include {@link Curve} and {@link Surface}. *

* The sensitivity is expressed as an array, with one entry for each parameter in the {@code ParameterizedData}. * The sensitivity has no associated currency. *

* A single {@code UnitParameterSensitivity} represents the sensitivity to a single {@code ParameterizedData} instance. * However, a {@code ParameterizedData} instance can itself be backed by more than one underlying instance. * For example, a curve formed from two underlying curves. * Information about the split between these underlying instances can optionally be stored. */ @BeanDefinition(builderScope = "private") public final class UnitParameterSensitivity implements ImmutableBean, Serializable { /** * The market data name. *

* This name is used in the market data system to identify the data that the sensitivities refer to. */ @PropertyDefinition(validate = "notNull") private final MarketDataName marketDataName; /** * The list of parameter metadata. *

* There is one entry for each parameter. */ @PropertyDefinition(validate = "notNull", builderType = "List") private final ImmutableList parameterMetadata; /** * The parameter sensitivity values. *

* There is one sensitivity value for each parameter. */ @PropertyDefinition(validate = "notNull") private final DoubleArray sensitivity; /** * The split of parameters between the underlying parameterized data. *

* A single {@code UnitParameterSensitivity} represents the sensitivity to a single {@link ParameterizedData} instance. * However, a {@code ParameterizedData} instance can itself be backed by more than one underlying instance. * For example, a curve formed from two underlying curves. * This list is present, it represents how to split this sensitivity between the underlying instances. */ @PropertyDefinition(get = "optional", type = "List<>") private final ImmutableList parameterSplit; //------------------------------------------------------------------------- /** * Obtains an instance from the market data name, metadata and sensitivity. *

* The market data name identifies the {@link ParameterizedData} instance that was queried. * The parameter metadata provides information on each parameter. * The size of the parameter metadata list must match the size of the sensitivity array. * * @param marketDataName the name of the market data that the sensitivity refers to * @param parameterMetadata the parameter metadata * @param sensitivity the sensitivity values, one for each parameter * @return the sensitivity object */ public static UnitParameterSensitivity of( MarketDataName marketDataName, List parameterMetadata, DoubleArray sensitivity) { return new UnitParameterSensitivity(marketDataName, parameterMetadata, sensitivity, null); } /** * Obtains an instance from the market data name and sensitivity. *

* The market data name identifies the {@link ParameterizedData} instance that was queried. * The parameter metadata will be empty. * The size of the parameter metadata list must match the size of the sensitivity array. * * @param marketDataName the name of the market data that the sensitivity refers to * @param sensitivity the sensitivity values, one for each parameter * @return the sensitivity object */ public static UnitParameterSensitivity of(MarketDataName marketDataName, DoubleArray sensitivity) { return of(marketDataName, ParameterMetadata.listOfEmpty(sensitivity.size()), sensitivity); } /** * Obtains an instance from the market data name, metadata, sensitivity and parameter split. *

* The market data name identifies the {@link ParameterizedData} instance that was queried. * The parameter metadata provides information on each parameter. * The size of the parameter metadata list must match the size of the sensitivity array. *

* The parameter split allows the sensitivity to represent the split between two or more * underlying {@link ParameterizedData} instances. The sum of the parameters in the split * must equal the size of the sensitivity array, and each name must be unique. * * @param marketDataName the name of the market data that the sensitivity refers to * @param parameterMetadata the parameter metadata * @param sensitivity the sensitivity values, one for each parameter * @param parameterSplit the split between the underlying {@code ParameterizedData} instances * @return the sensitivity object */ public static UnitParameterSensitivity of( MarketDataName marketDataName, List parameterMetadata, DoubleArray sensitivity, List parameterSplit) { return new UnitParameterSensitivity(marketDataName, parameterMetadata, sensitivity, parameterSplit); } //------------------------------------------------------------------------- /** * Combines two or more instances to form a single sensitivity instance. *

* The result will store information about the separate instances allowing it to be {@link #split()} later. * * @param marketDataName the combined name of the market data that the sensitivity refers to * @param sensitivities the sensitivity instances to combine, two or more * @return the combined sensitivity object */ public static UnitParameterSensitivity combine( MarketDataName marketDataName, UnitParameterSensitivity... sensitivities) { ArgChecker.notEmpty(sensitivities, "sensitivities"); if (sensitivities.length < 2) { throw new IllegalArgumentException("At least two sensitivity instances must be specified"); } int size = Stream.of(sensitivities).mapToInt(s -> s.getParameterCount()).sum(); double[] combinedSensitivities = new double[size]; ImmutableList.Builder combinedMeta = ImmutableList.builder(); ImmutableList.Builder split = ImmutableList.builder(); int count = 0; for (int i = 0; i < sensitivities.length; i++) { UnitParameterSensitivity sens = sensitivities[i]; System.arraycopy(sens.getSensitivity().toArrayUnsafe(), 0, combinedSensitivities, count, sens.getParameterCount()); combinedMeta.addAll(sens.getParameterMetadata()); split.add(ParameterSize.of(sens.getMarketDataName(), sens.getParameterCount())); count += sens.getParameterCount(); } return new UnitParameterSensitivity( marketDataName, combinedMeta.build(), DoubleArray.ofUnsafe(combinedSensitivities), split.build()); } @ImmutableValidator private void validate() { if (sensitivity.size() != parameterMetadata.size()) { throw new IllegalArgumentException("Length of sensitivity and parameter metadata must match"); } if (parameterSplit != null) { long total = parameterSplit.stream().mapToInt(p -> p.getParameterCount()).sum(); if (sensitivity.size() != total) { throw new IllegalArgumentException("Length of sensitivity and parameter split must match"); } if (parameterSplit.stream().map(p -> p.getName()).distinct().count() != parameterSplit.size()) { throw new IllegalArgumentException("Parameter split must not contain duplicate market data names"); } } } //------------------------------------------------------------------------- /** * Gets the number of parameters. *

* This returns the number of parameters in the {@link ParameterizedData} instance * which is the same size as the sensitivity array. * * @return the number of parameters */ public int getParameterCount() { return sensitivity.size(); } /** * Gets the parameter metadata at the specified index. *

* If there is no specific parameter metadata, an empty instance will be returned. * * @param parameterIndex the zero-based index of the parameter to get * @return the metadata of the parameter * @throws IndexOutOfBoundsException if the index is invalid */ public ParameterMetadata getParameterMetadata(int parameterIndex) { return parameterMetadata.get(parameterIndex); } /** * Compares the key of two sensitivity objects, excluding the parameter sensitivity values. * * @param other the other sensitivity object * @return positive if greater, zero if equal, negative if less */ public int compareKey(UnitParameterSensitivity other) { return ComparisonChain.start() .compare(marketDataName, other.marketDataName) .result(); } //------------------------------------------------------------------------- /** * Returns an instance converted this sensitivity to a monetary value, multiplying by the specified factor. *

* Each value in the sensitivity array will be multiplied by the specified factor. * * @param currency the currency of the amount * @param amount the amount to multiply by * @return the resulting sensitivity object */ public CurrencyParameterSensitivity multipliedBy(Currency currency, double amount) { return CurrencyParameterSensitivity.of( marketDataName, parameterMetadata, currency, sensitivity.multipliedBy(amount), parameterSplit); } /** * Returns an instance with the sensitivity values multiplied by the specified factor. *

* Each value in the sensitivity array will be multiplied by the factor. * * @param factor the multiplicative factor * @return an instance based on this one, with each sensitivity multiplied by the factor */ public UnitParameterSensitivity multipliedBy(double factor) { return mapSensitivity(s -> s * factor); } /** * Returns an instance with the specified operation applied to the sensitivity values. *

* Each value in the sensitivity array will be operated on. * For example, the operator could multiply the sensitivities by a constant, or take the inverse. *

   *   inverse = base.mapSensitivity(value -> 1 / value);
   * 
* * @param operator the operator to be applied to the sensitivities * @return an instance based on this one, with the operator applied to the sensitivity values */ public UnitParameterSensitivity mapSensitivity(DoubleUnaryOperator operator) { return new UnitParameterSensitivity(marketDataName, parameterMetadata, sensitivity.map(operator), parameterSplit); } /** * Returns an instance with new parameter sensitivity values. * * @param sensitivity the new sensitivity values * @return an instance based on this one, with the specified sensitivity values */ public UnitParameterSensitivity withSensitivity(DoubleArray sensitivity) { return new UnitParameterSensitivity(marketDataName, parameterMetadata, sensitivity, parameterSplit); } /** * Returns an instance with the specified sensitivity array added to the array in this instance. *

* The specified array must match the size of the array in this instance. * * @param otherSensitivty the other parameter sensitivity * @return an instance based on this one, with the other instance added * @throws IllegalArgumentException if the market data name, metadata or parameter split differs */ public UnitParameterSensitivity plus(DoubleArray otherSensitivty) { if (otherSensitivty.size() != sensitivity.size()) { throw new IllegalArgumentException(Messages.format( "Sensitivity array size {} must match size {}", otherSensitivty.size(), sensitivity.size())); } return withSensitivity(sensitivity.plus(otherSensitivty)); } /** * Returns an instance with the specified sensitivity array added to the array in this instance. *

* The specified instance must have the same name, metadata and parameter split as this instance. * * @param otherSensitivty the other parameter sensitivity * @return an instance based on this one, with the other instance added * @throws IllegalArgumentException if the market data name, metadata or parameter split differs */ public UnitParameterSensitivity plus(UnitParameterSensitivity otherSensitivty) { if (!marketDataName.equals(otherSensitivty.marketDataName) || !parameterMetadata.equals(otherSensitivty.parameterMetadata) || (parameterSplit != null && !parameterSplit.equals(otherSensitivty.parameterSplit))) { throw new IllegalArgumentException("Two sensitivity instances can only be added if name, metadata and split are equal"); } return plus(otherSensitivty.getSensitivity()); } //------------------------------------------------------------------------- /** * Splits this sensitivity instance. *

* A single sensitivity instance may be based on more than one underlying {@link ParameterizedData}, * as represented by {@link #getParameterSplit()}. Calling this method returns a list * where the sensitivity of this instance has been split into multiple instances as per * the parameter split definition. In the common case where there is a single underlying * {@code ParameterizedData}, the list will be of size one containing this instance. * * @return this sensitivity split as per the defined parameter split, ordered as per this instance */ public ImmutableList split() { if (parameterSplit == null) { return ImmutableList.of(this); } ImmutableList.Builder builder = ImmutableList.builder(); int count = 0; for (ParameterSize size : parameterSplit) { List splitMetadata = parameterMetadata.subList(count, count + size.getParameterCount()); DoubleArray splitSensitivity = sensitivity.subArray(count, count + size.getParameterCount()); builder.add(UnitParameterSensitivity.of(size.getName(), splitMetadata, splitSensitivity)); count += size.getParameterCount(); } return builder.build(); } //------------------------------------------------------------------------- /** * Returns the total of the sensitivity values. * * @return the total sensitivity values */ public double total() { return sensitivity.sum(); } //------------------------- AUTOGENERATED START ------------------------- /** * The meta-bean for {@code UnitParameterSensitivity}. * @return the meta-bean, not null */ public static UnitParameterSensitivity.Meta meta() { return UnitParameterSensitivity.Meta.INSTANCE; } static { MetaBean.register(UnitParameterSensitivity.Meta.INSTANCE); } /** * The serialization version id. */ private static final long serialVersionUID = 1L; private UnitParameterSensitivity( MarketDataName marketDataName, List parameterMetadata, DoubleArray sensitivity, List parameterSplit) { JodaBeanUtils.notNull(marketDataName, "marketDataName"); JodaBeanUtils.notNull(parameterMetadata, "parameterMetadata"); JodaBeanUtils.notNull(sensitivity, "sensitivity"); this.marketDataName = marketDataName; this.parameterMetadata = ImmutableList.copyOf(parameterMetadata); this.sensitivity = sensitivity; this.parameterSplit = (parameterSplit != null ? ImmutableList.copyOf(parameterSplit) : null); validate(); } @Override public UnitParameterSensitivity.Meta metaBean() { return UnitParameterSensitivity.Meta.INSTANCE; } //----------------------------------------------------------------------- /** * Gets the market data name. *

* This name is used in the market data system to identify the data that the sensitivities refer to. * @return the value of the property, not null */ public MarketDataName getMarketDataName() { return marketDataName; } //----------------------------------------------------------------------- /** * Gets the list of parameter metadata. *

* There is one entry for each parameter. * @return the value of the property, not null */ public ImmutableList getParameterMetadata() { return parameterMetadata; } //----------------------------------------------------------------------- /** * Gets the parameter sensitivity values. *

* There is one sensitivity value for each parameter. * @return the value of the property, not null */ public DoubleArray getSensitivity() { return sensitivity; } //----------------------------------------------------------------------- /** * Gets the split of parameters between the underlying parameterized data. *

* A single {@code UnitParameterSensitivity} represents the sensitivity to a single {@link ParameterizedData} instance. * However, a {@code ParameterizedData} instance can itself be backed by more than one underlying instance. * For example, a curve formed from two underlying curves. * This list is present, it represents how to split this sensitivity between the underlying instances. * @return the optional value of the property, not null */ public Optional> getParameterSplit() { return Optional.ofNullable(parameterSplit); } //----------------------------------------------------------------------- @Override public boolean equals(Object obj) { if (obj == this) { return true; } if (obj != null && obj.getClass() == this.getClass()) { UnitParameterSensitivity other = (UnitParameterSensitivity) obj; return JodaBeanUtils.equal(marketDataName, other.marketDataName) && JodaBeanUtils.equal(parameterMetadata, other.parameterMetadata) && JodaBeanUtils.equal(sensitivity, other.sensitivity) && JodaBeanUtils.equal(parameterSplit, other.parameterSplit); } return false; } @Override public int hashCode() { int hash = getClass().hashCode(); hash = hash * 31 + JodaBeanUtils.hashCode(marketDataName); hash = hash * 31 + JodaBeanUtils.hashCode(parameterMetadata); hash = hash * 31 + JodaBeanUtils.hashCode(sensitivity); hash = hash * 31 + JodaBeanUtils.hashCode(parameterSplit); return hash; } @Override public String toString() { StringBuilder buf = new StringBuilder(160); buf.append("UnitParameterSensitivity{"); buf.append("marketDataName").append('=').append(marketDataName).append(',').append(' '); buf.append("parameterMetadata").append('=').append(parameterMetadata).append(',').append(' '); buf.append("sensitivity").append('=').append(sensitivity).append(',').append(' '); buf.append("parameterSplit").append('=').append(JodaBeanUtils.toString(parameterSplit)); buf.append('}'); return buf.toString(); } //----------------------------------------------------------------------- /** * The meta-bean for {@code UnitParameterSensitivity}. */ public static final class Meta extends DirectMetaBean { /** * The singleton instance of the meta-bean. */ static final Meta INSTANCE = new Meta(); /** * The meta-property for the {@code marketDataName} property. */ @SuppressWarnings({"unchecked", "rawtypes" }) private final MetaProperty> marketDataName = DirectMetaProperty.ofImmutable( this, "marketDataName", UnitParameterSensitivity.class, (Class) MarketDataName.class); /** * The meta-property for the {@code parameterMetadata} property. */ @SuppressWarnings({"unchecked", "rawtypes" }) private final MetaProperty> parameterMetadata = DirectMetaProperty.ofImmutable( this, "parameterMetadata", UnitParameterSensitivity.class, (Class) ImmutableList.class); /** * The meta-property for the {@code sensitivity} property. */ private final MetaProperty sensitivity = DirectMetaProperty.ofImmutable( this, "sensitivity", UnitParameterSensitivity.class, DoubleArray.class); /** * The meta-property for the {@code parameterSplit} property. */ @SuppressWarnings({"unchecked", "rawtypes" }) private final MetaProperty> parameterSplit = DirectMetaProperty.ofImmutable( this, "parameterSplit", UnitParameterSensitivity.class, (Class) List.class); /** * The meta-properties. */ private final Map> metaPropertyMap$ = new DirectMetaPropertyMap( this, null, "marketDataName", "parameterMetadata", "sensitivity", "parameterSplit"); /** * Restricted constructor. */ private Meta() { } @Override protected MetaProperty metaPropertyGet(String propertyName) { switch (propertyName.hashCode()) { case 842855857: // marketDataName return marketDataName; case -1169106440: // parameterMetadata return parameterMetadata; case 564403871: // sensitivity return sensitivity; case 1122130161: // parameterSplit return parameterSplit; } return super.metaPropertyGet(propertyName); } @Override public BeanBuilder builder() { return new UnitParameterSensitivity.Builder(); } @Override public Class beanType() { return UnitParameterSensitivity.class; } @Override public Map> metaPropertyMap() { return metaPropertyMap$; } //----------------------------------------------------------------------- /** * The meta-property for the {@code marketDataName} property. * @return the meta-property, not null */ public MetaProperty> marketDataName() { return marketDataName; } /** * The meta-property for the {@code parameterMetadata} property. * @return the meta-property, not null */ public MetaProperty> parameterMetadata() { return parameterMetadata; } /** * The meta-property for the {@code sensitivity} property. * @return the meta-property, not null */ public MetaProperty sensitivity() { return sensitivity; } /** * The meta-property for the {@code parameterSplit} property. * @return the meta-property, not null */ public MetaProperty> parameterSplit() { return parameterSplit; } //----------------------------------------------------------------------- @Override protected Object propertyGet(Bean bean, String propertyName, boolean quiet) { switch (propertyName.hashCode()) { case 842855857: // marketDataName return ((UnitParameterSensitivity) bean).getMarketDataName(); case -1169106440: // parameterMetadata return ((UnitParameterSensitivity) bean).getParameterMetadata(); case 564403871: // sensitivity return ((UnitParameterSensitivity) bean).getSensitivity(); case 1122130161: // parameterSplit return ((UnitParameterSensitivity) bean).parameterSplit; } return super.propertyGet(bean, propertyName, quiet); } @Override protected void propertySet(Bean bean, String propertyName, Object newValue, boolean quiet) { metaProperty(propertyName); if (quiet) { return; } throw new UnsupportedOperationException("Property cannot be written: " + propertyName); } } //----------------------------------------------------------------------- /** * The bean-builder for {@code UnitParameterSensitivity}. */ private static final class Builder extends DirectPrivateBeanBuilder { private MarketDataName marketDataName; private List parameterMetadata = ImmutableList.of(); private DoubleArray sensitivity; private List parameterSplit; /** * Restricted constructor. */ private Builder() { } //----------------------------------------------------------------------- @Override public Object get(String propertyName) { switch (propertyName.hashCode()) { case 842855857: // marketDataName return marketDataName; case -1169106440: // parameterMetadata return parameterMetadata; case 564403871: // sensitivity return sensitivity; case 1122130161: // parameterSplit return parameterSplit; default: throw new NoSuchElementException("Unknown property: " + propertyName); } } @SuppressWarnings("unchecked") @Override public Builder set(String propertyName, Object newValue) { switch (propertyName.hashCode()) { case 842855857: // marketDataName this.marketDataName = (MarketDataName) newValue; break; case -1169106440: // parameterMetadata this.parameterMetadata = (List) newValue; break; case 564403871: // sensitivity this.sensitivity = (DoubleArray) newValue; break; case 1122130161: // parameterSplit this.parameterSplit = (List) newValue; break; default: throw new NoSuchElementException("Unknown property: " + propertyName); } return this; } @Override public UnitParameterSensitivity build() { return new UnitParameterSensitivity( marketDataName, parameterMetadata, sensitivity, parameterSplit); } //----------------------------------------------------------------------- @Override public String toString() { StringBuilder buf = new StringBuilder(160); buf.append("UnitParameterSensitivity.Builder{"); buf.append("marketDataName").append('=').append(JodaBeanUtils.toString(marketDataName)).append(',').append(' '); buf.append("parameterMetadata").append('=').append(JodaBeanUtils.toString(parameterMetadata)).append(',').append(' '); buf.append("sensitivity").append('=').append(JodaBeanUtils.toString(sensitivity)).append(',').append(' '); buf.append("parameterSplit").append('=').append(JodaBeanUtils.toString(parameterSplit)); buf.append('}'); return buf.toString(); } } //-------------------------- AUTOGENERATED END -------------------------- }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy