Ejemplo n.º 1
0
    def test_find_interval_index_one_interval(self):
        """Tests find_interval_index is correct with one half-open interval."""
        result = self.evaluate(piecewise.find_interval_index([1.0], [1.0]))
        self.assertAllEqual(result, [0])

        result = self.evaluate(piecewise.find_interval_index([0.0], [1.0]))
        self.assertAllEqual(result, [-1])

        result = self.evaluate(piecewise.find_interval_index([2.0], [1.0]))
        self.assertAllEqual(result, [0])
Ejemplo n.º 2
0
def _linear_interpolation_single_batch(x, x_data, y_data, left_slope,
                                       right_slope, validate_args):
  """Calculates the result of the linear interpolation for a single batch."""

  control_deps = []
  if validate_args:
    control_deps.append(tf.Assert(tf.math.is_non_decreasing(x_data), [x_data]))

  with tf.control_dependencies(control_deps):
    # Get upper bound indices for `x`.
    upper_indices = piecewise.find_interval_index(x, x_data)
    x_data_size = tf.shape(x_data)[-1]
    at_min = tf.equal(upper_indices, -1)
    at_max = tf.equal(upper_indices, x_data_size - 1)

    # Create tensors in order to be used by `tf.where`.
    # `values_min` are extrapolated values for x-coordinates less than or
    # equal to `x_data[0]`.
    # `values_max` are extrapolated values for x-coordinates greater than
    # `x_data[-1]`.
    values_min = y_data[0] + left_slope * (
        x - tf.broadcast_to(x_data[0], shape=tf.shape(x)))
    values_max = y_data[-1] + right_slope * (
        x - tf.broadcast_to(x_data[-1], shape=tf.shape(x)))

    # `tf.where` evaluates all branches, need to cap indices to ensure it
    # won't go out of bounds.
    capped_lower_indices = tf.math.maximum(upper_indices, 0)
    capped_upper_indices = tf.math.minimum(upper_indices + 1, x_data_size - 1)
    x_data_lower = tf.gather(x_data, capped_lower_indices)
    x_data_upper = tf.gather(x_data, capped_upper_indices)
    y_data_lower = tf.gather(y_data, capped_lower_indices)
    y_data_upper = tf.gather(y_data, capped_upper_indices)

    # Nan in unselected branches could propagate through gradient calculation,
    # hence we need to clip the values to ensure no nan would occur. In this
    # case we need to ensure there is no division by zero.
    x_data_diff = x_data_upper - x_data_lower
    floor_x_diff = tf.where(at_min | at_max, x_data_diff + 1, x_data_diff)
    interpolated = y_data_lower + (x - x_data_lower) * (
        y_data_upper - y_data_lower) / floor_x_diff

    interpolated = tf.where(at_min, values_min, interpolated)
    interpolated = tf.where(at_max, values_max, interpolated)

    return interpolated
Ejemplo n.º 3
0
 def test_find_interval_index_last_interval_is_closed(self):
     """Tests find_interval_index is correct in the general case."""
     result = piecewise.find_interval_index([3.0, 4.0], [2.0, 3.0],
                                            last_interval_is_closed=True)
     self.assertAllEqual(result, [0, 1])
Ejemplo n.º 4
0
 def test_find_interval_index(self):
     """Tests find_interval_index is correct in the general case."""
     interval_lower_xs = [0.25, 0.5, 1.0, 2.0, 3.0]
     query_xs = [0.25, 3.0, 5.0, 0.0, 0.5, 0.8]
     result = piecewise.find_interval_index(query_xs, interval_lower_xs)
     self.assertAllEqual(result, [0, 4, 4, -1, 1, 1])
Ejemplo n.º 5
0
 def test_find_interval_index_correct_dtype(self):
     """Tests find_interval_index outputs the correct type."""
     result = self.evaluate(piecewise.find_interval_index([1.0],
                                                          [0.0, 1.0]))
     self.assertIsInstance(result[0], np.int32)
Ejemplo n.º 6
0
def interpolate(times,
                interval_values,
                interval_times,
                validate_args=False,
                dtype=None,
                name=None):
    """Performs the monotone convex interpolation.

  The monotone convex method is a scheme devised by Hagan and West (Ref [1]). It
  is a commonly used method to interpolate interest rate yield curves. For
  more details see Refs [1, 2].

  It is important to point out that the monotone convex method *does not* solve
  the standard interpolation problem but a modified one as described below.

  Suppose we are given a strictly increasing sequence of scalars (which we will
  refer to as time) `[t_1, t_2, ... t_n]` and a set of values
  `[f_1, f_2, ... f_n]`.
  The aim is to find a function `f(t)` defined on the interval `[0, t_n]` which
  satisfies (in addition to continuity and positivity conditions) the following

  ```None
    Integral[f(u), t_{i-1} <= u <= t_i] = f_i,  with t_0 = 0

  ```

  In the context of interest rate curve building, `f(t)` corresponds to the
  instantaneous forward rate at time `t` and the `f_i` correspond to the
  discrete forward rates that apply to the time period `[t_{i-1}, t_i]`.
  Furthermore, the integral of the forward curve is related to the yield curve
  by

  ```None
    Integral[f(u), 0 <= u <= t] = r(t) * t

  ```

  where `r(t)` is the interest rate that applies between `[0, t]` (the yield of
  a zero coupon bond paying a unit of currency at time `t`).

  This function computes both the interpolated value and the integral along
  the segment containing the supplied time. Specifically, given a time `t` such
  that `t_k <= t <= t_{k+1}`, this function computes the interpolated value
  `f(t)` and the value `Integral[f(u), t_k <= u <= t]`.

  This implementation of the method currently supports batching along the
  interpolation times but not along the interpolated curves (i.e. it is possible
  to evaluate the `f(t)` for `t` as a vector of times but not build multiple
  curves at the same time).

  #### Example

  ```python
  interval_times = tf.constant([0.25, 0.5, 1.0, 2.0, 3.0], dtype=dtype)
  interval_values = tf.constant([0.05, 0.051, 0.052, 0.053, 0.055],
                                dtype=dtype)
  times = tf.constant([0.25, 0.5, 1.0, 2.0, 3.0, 1.1], dtype=dtype)
  # Returns the following two values:
  # interpolated = [0.0505, 0.05133333, 0.05233333, 0.054, 0.0555, 0.05241]
  # integrated =  [0, 0, 0, 0, 0.055, 0.005237]
  # Note that the first four integrated values are zero. This is because
  # those interpolation time are at the start of their containing interval.
  # The fourth value (i.e. at 3.0) is not zero because this is the last
  # interval (i.e. it is the integral from 2.0 to 3.0).
  interpolated, integrated = interpolate(
      times, interval_values, interval_times)
  ```

  #### References:

  [1]: Patrick Hagan & Graeme West. Interpolation Methods for Curve
    Construction. Applied Mathematical Finance. Vol 13, No. 2, pp 89-129.
    June 2006.
    https://www.researchgate.net/publication/24071726_Interpolation_Methods_for_Curve_Construction
  [2]: Patrick Hagan & Graeme West. Methods for Constructing a Yield Curve.
    Wilmott Magazine, pp. 70-81. May 2008.

  Args:
    times: Non-negative rank 1 `Tensor` of any size. The times for which the
      interpolation has to be performed.
    interval_values: Rank 1 `Tensor` of the same shape and dtype as
      `interval_times`. The values associated to each of the intervals specified
      by the `interval_times`. Must have size at least 2.
    interval_times: Strictly positive rank 1 `Tensor` of real dtype containing
      increasing values. The endpoints of the intervals (i.e. `t_i` above.).
      Note that the left end point of the first interval is implicitly assumed
      to be 0. Must have size at least 2.
    validate_args: Python bool. If true, adds control dependencies to check that
      the `times` are bounded by the `interval_endpoints`.
      Default value: False
    dtype: `tf.Dtype` to use when converting arguments to `Tensor`s. If not
      supplied, the default Tensorflow conversion will take place. Note that
      this argument does not do any casting.
      Default value: None.
    name: Python `str` name prefixed to Ops created by this class.
      Default value: None which is mapped to the default name 'interpolation'.

  Returns:
    A 2-tuple containing
      interpolated_values: Rank 1 `Tensor` of the same size and dtype as the
        `times`. The interpolated values at the supplied times.
      integrated_values: Rank 1 `Tensor` of the same size and dtype as the
        `times`. The integral of the interpolated function. The integral is
        computed from the largest interval time that is smaller than the time
        up to the given time.
  """
    with tf.compat.v1.name_scope(
            name,
            default_name='interpolate',
            values=[times, interval_times, interval_values]):
        times = tf.convert_to_tensor(times, dtype=dtype, name='times')
        interval_times = tf.convert_to_tensor(interval_times,
                                              dtype=dtype,
                                              name='interval_times')
        interval_values = tf.convert_to_tensor(interval_values,
                                               dtype=dtype,
                                               name='interval_values')
        control_deps = []
        if validate_args:
            control_deps = [
                tf.compat.v1.debugging.assert_non_negative(times),
                tf.compat.v1.debugging.assert_positive(interval_times)
            ]
        with tf.compat.v1.control_dependencies(control_deps):
            # Step 1: Find the values at the endpoints.
            endpoint_values = _interpolate_adjacent(interval_times,
                                                    interval_values)
            endpoint_times = tf.concat([[0.0], interval_times], axis=0)
            intervals = piecewise.find_interval_index(
                times, endpoint_times, last_interval_is_closed=True)
            # Comparing to the notation used in the paper:
            # f_left -> f_{i-1}
            # f_right -> f_i
            # t_left -> t_{i-1}
            # t_right -> t_i
            # fd -> f^d_i
            # g0 -> g0
            # g1 -> g1
            # g1plus2g0 -> g1 + 2 g0 (boundary line A)
            # g0plus2g1 -> g0 + 2 g1 (boundary line B)
            # x -> x
            f_left = tf.gather(endpoint_values, intervals)
            f_right = tf.gather(endpoint_values, intervals + 1)
            # fd is the discrete forward associated to each interval.
            fd = tf.gather(interval_values, intervals)
            t_left = tf.gather(endpoint_times, intervals)
            t_right = tf.gather(endpoint_times, intervals + 1)
            interval_lengths = (t_right - t_left)
            x = (times - t_left) / interval_lengths

            # TODO(b/140410758): The calculation below can be done more efficiently
            # if we instead do the following:
            # 1. Subdivide the regions further so that each subregion corresponds
            #   to a single quadratic in x. (Region 2, 3 and 4 get divided into 2
            #   pieces for a total of 7 cases.
            # 2. For each interval (i.e. [t_i, t{i+1}]) the case that applies to
            #   a point falling in that region can be decided and the corresponding
            #   quadratic coefficients computed once and for all.
            # 3. The above information is built once for the supplied forwards.
            # 4. Given the above information and a set of times to interpolate for,
            #   we map each time to the appropriate interval and compute the quadratic
            #   function value using that x.

            g0 = f_left - fd
            g1 = f_right - fd
            g1plus2g0 = g1 + 2 * g0
            g0plus2g1 = g0 + 2 * g1

            result = tf.zeros_like(times)
            integrated = tf.zeros_like(times)

            # The method uses quadratic splines to do the interpolation.
            # The specific spline used depends on the relationship between the
            # boundary values (`g0` and `g1` above).
            # The two dimensional plane determined by these two values is divided
            # into four wedge sections referred to as region 1, 2, 3 and 4 below.
            # For details of how the regions are defined, see Fig. 4 in Ref [2].
            is_region_1, region_1_value, integrated_value_1 = _region_1(
                g1plus2g0, g0plus2g1, g0, g1, x)

            result = tf.where(is_region_1, region_1_value, result)
            integrated = tf.where(is_region_1, integrated_value_1, integrated)

            is_region_2, region_2_value, integrated_value_2 = _region_2(
                g1plus2g0, g0plus2g1, g0, g1, x)

            result = tf.where(is_region_2, region_2_value, result)
            integrated = tf.where(is_region_2, integrated_value_2, integrated)

            is_region_3, region_3_value, integrated_value_3 = _region_3(
                g1plus2g0, g0plus2g1, g0, g1, x)

            result = tf.where(is_region_3, region_3_value, result)
            integrated = tf.where(is_region_3, integrated_value_3, integrated)

            is_region_4, region_4_value, integrated_value_4 = _region_4(
                g1plus2g0, g0plus2g1, g0, g1, x)
            result = tf.where(is_region_4, region_4_value, result)
            integrated = tf.where(is_region_4, integrated_value_4, integrated)

            # g0 = g1 = 0 requires special handling. Checking if the values are
            # legitimatey zero requires we pay close attention to the numerical
            # precision issues.
            g0_eps = tf.abs(tf.math.nextafter(fd, f_left) - fd) * 1.1
            g1_eps = tf.abs(tf.math.nextafter(fd, f_right) - fd) * 1.1

            is_origin = ((tf.abs(g0) <= g0_eps) & (tf.abs(g1) <= g1_eps))

            result = tf.where(is_origin, tf.zeros_like(result), result)
            integrated = tf.where(is_origin, tf.zeros_like(integrated),
                                  integrated)

            return (result + fd, (integrated + fd * x) * interval_lengths)
Ejemplo n.º 7
0
def interpolate_yields(interpolation_times,
                       reference_times,
                       yields=None,
                       discrete_forwards=None,
                       validate_args=False,
                       dtype=None,
                       name=None):
    """Interpolates the yield curve to the supplied times.

    Applies the Hagan West procedure to interpolate either a zero coupon yield
    curve or a discrete forward curve to a given set of times.
    A zero coupon yield curve is specified by a set
    of times and the yields on zero coupon bonds expiring at those
    times. A discrete forward rate curve specifies the interest rate that
    applies between two times in the future as seen from the current time.
    The relation between the two sets of curve is as follows. Suppose the
    yields on zero coupon bonds expiring at times `[t_1, ..., t_n]` are
    `[r_1, ..., r_n]`, then the forward rate between time `[t_i, t_{i+1}]` is
    denoted `f(0; t_i, t_{i+1})` and given by

    ```None
      f(0; t_i, t_{i+1}) = (r_{i+1} t_{i+1} - r_i t_i) / (t_{i+1} - t_i)
    ```

    This function uses the Hagan West algorithm to perform the interpolation.
    The interpolation proceeds in two steps. Firstly the discrete forward
    curve is bootstrapped and an instantaneous forward curve is built. From the
    instantaneous forward curve, the interpolated yield values are inferred
    using the relation:

    ```None
      r(t) = (1/t) * Integrate[ f(s), 0 <= s <= t]
    ```

    The above equation connects the instantaneous forward curve `f(t)` to the
    yield curve `r(t)`. The Hagan West procedure uses the Monotone Convex
    interpolation to create a continuous forward curve. This is then integrated
    to compute the implied yield rate.

    For more details on the interpolation procedure, see Ref. [1].

  #### Example

  ```python
    dtype = np.float64
    reference_times = np.array([1.0, 2.0, 3.0, 4.0], dtype=dtype)
    yields = np.array([5.0, 4.75, 4.53333333, 4.775], dtype=dtype)

    # Times for which the interpolated values are required.
    interpolation_times = np.array([0.25, 0.5, 1.0, 2.0], dtype=dtype)

    interpolated = interpolate_yields(
        interpolation_times, reference_times, yields=yields)
    # Produces [5.1171875, 5.09375, 5.0, 4.75]
  ```

  #### References:

  [1]: Patrick Hagan & Graeme West. Methods for Constructing a Yield Curve.
    Wilmott Magazine, pp. 70-81. May 2008.
    https://www.researchgate.net/profile/Patrick_Hagan3/publication/228463045_Methods_for_constructing_a_yield_curve/links/54db8cda0cf23fe133ad4d01.pdf

  Args:
    interpolation_times: Non-negative rank 1 `Tensor` of any size. The times for
      which the interpolation has to be performed.
    reference_times: Strictly positive rank 1 `Tensor` of real dtype containing
      increasing values. The expiry times of the underlying zero coupon bonds.
    yields: Optional rank 1 `Tensor` of the same shape and dtype as
      `reference_times`, if supplied. The yield rate of zero coupon bonds
      expiring at the corresponding time in the `reference_times`. Either this
      argument or the `discrete_forwards` must be supplied (but not both).
      Default value: None.
    discrete_forwards: Optional rank 1 `Tensor` of the same shape and dtype as
      `reference_times`, if supplied. The `i`th component of the `Tensor` is the
      forward rate that applies between `reference_times[i-1]` and
      `reference_times[i]` for `i>0` and between time `0` and
      `reference_times[0]` for `i=0`. Either this argument or the `yields` must
      be specified (but not both).
      Default value: None.
    validate_args: Python bool. If true, adds control dependencies to check that
      the `times` are bounded by the `reference_times`.
      Default value: False
    dtype: `tf.Dtype` to use when converting arguments to `Tensor`s. If not
      supplied, the default Tensorflow conversion will take place. Note that
      this argument does not do any casting.
      Default value: None.
    name: Python `str` name prefixed to Ops created by this class.
      Default value: None which is mapped to the default name
        'interpolate_forward_rate'.

  Returns:
      interpolated_forwards: Rank 1 `Tensor` of the same size and dtype as the
        `interpolation_times`. The interpolated instantaneous forwards at the
        `interpolation_times`.

  Raises:
    ValueError if neither `yields` nor `discrete_forwards` are specified or if
    both are specified.
  """

    if (yields is None) == (discrete_forwards is None):
        raise ValueError('Exactly one of yields or discrete forwards must'
                         ' be supplied.')

    with tf.compat.v1.name_scope(name,
                                 default_name='interpolate_forward_rate',
                                 values=[
                                     interpolation_times, reference_times,
                                     yields, discrete_forwards
                                 ]):
        if discrete_forwards is not None:
            discrete_forwards = tf.convert_to_tensor(discrete_forwards,
                                                     dtype=dtype)
            reference_yields = forwards.yields_from_forward_rates(
                discrete_forwards, reference_times, dtype=dtype)
        reference_times = tf.convert_to_tensor(reference_times, dtype=dtype)
        interpolation_times = tf.convert_to_tensor(interpolation_times,
                                                   dtype=dtype)

        if yields is not None:
            reference_yields = tf.convert_to_tensor(yields, dtype=dtype)
            discrete_forwards = forwards.forward_rates_from_yields(
                reference_yields, reference_times, dtype=dtype)

        _, integrated_adjustments = interpolate(interpolation_times,
                                                discrete_forwards,
                                                reference_times,
                                                validate_args=validate_args,
                                                dtype=dtype)

        extended_times = tf.concat([[0.0], reference_times], axis=0)
        extended_yields = tf.concat([[0.0], reference_yields], axis=0)
        intervals = piecewise.find_interval_index(interpolation_times,
                                                  extended_times,
                                                  last_interval_is_closed=True)
        base_values = tf.gather(extended_yields * extended_times, intervals)
        interpolated = tf.math.divide_no_nan(
            base_values + integrated_adjustments, interpolation_times)
        return interpolated