36. Cass-Koopmans Model with Distorting Taxes#

36.1. Overview#

This lecture studies effects of foreseen fiscal and technology shocks on competitive equilibrium prices and quantities in a nonstochastic version of a Cass-Koopmans growth model with features described in this QuantEcon lecture Cass-Koopmans Competitive Equilibrium.

We use the model as a laboratory to experiment with numerical techniques for approximating equilibria and to display the structure of dynamic models in which decision makers have perfect foresight about future government decisions.

Following a classic paper by Robert E. Hall [Hall, 1971], we augment a nonstochastic version of the Cass-Koopmans optimal growth model with a government that purchases a stream of goods and that finances its purchases with an sequences of several distorting flat-rate taxes.

Distorting taxes prevent a competitive equilibrium allocation from solving a planning problem.

Therefore, to compute an equilibrium allocation and price system, we solve a system of nonlinear difference equations consisting of the first-order conditions for decision makers and the other equilibrium conditions.

We present two ways to approximate an equilibrium:

  • The first is a shooting algorithm like the one that we deployed in Cass-Koopmans Competitive Equilibrium.

  • The second method is a root-finding algorithm that minimizes residuals from the first-order conditions of a consumer and a representative firm.

36.2. The Economy#

36.2.1. Technology#

Feasible allocations satisfy

(36.1)#\[ g_t + c_t + x_t \leq F(k_t, n_t), \]

where

  • \(g_t\) is government purchases of the time \(t\) good

  • \(x_t\) is gross investment, and

  • \(F(k_t, n_t)\) is a linearly homogeneous production function with positive and decreasing marginal products of capital \(k_t\) and labor \(n_t\).

Physical capital evolves according to

\[ k_{t+1} = (1 - \delta)k_t + x_t, \]

where \(\delta \in (0, 1)\) is a depreciation rate.

It is sometimes convenient to eliminate \(x_t\) from (36.1) and to represent it as as

\[ g_t + c_t + k_{t+1} \leq F(k_t, n_t) + (1 - \delta)k_t. \]

36.2.2. Components of a competitive equilibrium#

All trades occurring at time \(0\).

The representative household owns capital, makes investment decisions, and rents capital and labor to a representative production firm.

The representative firm uses capital and labor to produce goods with the production function \(F(k_t, n_t)\).

A price system is a triple of sequences \(\{q_t, \eta_t, w_t\}_{t=0}^\infty\), where

  • \(q_t\) is the time \(0\) pretax price of one unit of investment or consumption at time \(t\) (\(x_t\) or \(c_t\)),

  • \(\eta_t\) is the pretax price at time \(t\) that the household receives from the firm for renting capital at time \(t\), and

  • \(w_t\) is the pretax price at time \(t\) that the household receives for renting labor to the firm at time \(t\).

The prices \(w_t\) and \(\eta_t\) are expressed in terms of time \(t\) goods, while \(q_t\) is expressed in terms of a numeraire at time \(0\), as in Cass-Koopmans Competitive Equilibrium.

The presence of a government distinguishes this lecture from Cass-Koopmans Competitive Equilibrium.

Government purchases of goods at time \(t\) are \(g_t \geq 0\).

A government expenditure plan is a sequence \(g = \{g_t\}_{t=0}^\infty\).

A government tax plan is a \(4\)-tuple of sequences \(\{\tau_{ct}, \tau_{kt}, \tau_{nt}, \tau_{ht}\}_{t=0}^\infty\), where

  • \(\tau_{ct}\) is a tax rate on consumption at time \(t\),

  • \(\tau_{kt}\) is a tax rate on rentals of capital at time \(t\),

  • \(\tau_{nt}\) is a tax rate on wage earnings at time \(t\), and

  • \(\tau_{ht}\) is a lump sum tax on a consumer at time \(t\).

Because lump-sum taxes \(\tau_{ht}\) are available, the government actually should not use any distorting taxes.

Nevertheless, we include all of these taxes because, like [Hall, 1971], they allow us to analyze how various taxes distort production and consumption decisions.

In the experiment section, we shall see how variations in government tax plan affect the transition path and equilibrium.

36.2.3. Representative Household#

A representative household has preferences over nonnegative streams of a single consumption good \(c_t\) and leisure \(1-n_t\) that are ordered by:

(36.2)#\[ \sum_{t=0}^{\infty} \beta^t U(c_t, 1-n_t), \quad \beta \in (0, 1), \]

where

  • \(U\) is strictly increasing in \(c_t\), twice continuously differentiable, and strictly concave with \(c_t \geq 0\) and \(n_t \in [0, 1]\).

The representative hßousehold maximizes (36.2) subject to the single budget constraint:

(36.3)#\[\begin{split} \begin{aligned} \sum_{t=0}^\infty& q_t \left\{ (1 + \tau_{ct})c_t + \underbrace{[k_{t+1} - (1 - \delta)k_t]}_{\text{no tax when investing}} \right\} \\ &\leq \sum_{t=0}^\infty q_t \left\{ \eta_t k_t - \underbrace{\tau_{kt}(\eta_t - \delta)k_t}_{\text{tax on rental return}} + (1 - \tau_{nt})w_t n_t - \tau_{ht} \right\}. \end{aligned} \end{split}\]

Here we have assumed that the government gives a depreciation allowance \(\delta k_t\) from the gross rentals on capital \(\eta_t k_t\) and so collects taxes \(\tau_{kt} (\eta_t - \delta) k_t\) on rentals from capital.

36.2.4. Government#

Government plans \(\{ g_t \}_{t=0}^\infty\) for government purchases and taxes \(\{\tau_{ct}, \tau_{kt}, \tau_{nt}, \tau_{ht}\}_{t=0}^\infty\) must respect the budget constraint

(36.4)#\[ \sum_{t=0}^\infty q_t g_t \leq \sum_{t=0}^\infty q_t \left\{ \tau_{ct}c_t + \tau_{kt}(\eta_t - \delta)k_t + \tau_{nt}w_t n_t + \tau_{ht} \right\}. \]

Given a budget-feasible government policy \(\{g_t\}_{t=0}^\infty\) and \(\{\tau_{ct}, \tau_{kt}, \tau_{nt}, \tau_{ht}\}_{t=0}^\infty\) subject to (36.4),

  • Household chooses \(\{c_t\}_{t=0}^\infty\), \(\{n_t\}_{t=0}^\infty\), and \(\{k_{t+1}\}_{t=0}^\infty\) to maximize utility(36.2) subject to budget constraint(36.3), and

  • Frim chooses sequences of capital \(\{k_t\}_{t=0}^\infty\) and \(\{n_t\}_{t=0}^\infty\) to maximize profits

    (36.5)#\[ \sum_{t=0}^\infty q_t [F(k_t, n_t) - \eta_t k_t - w_t n_t] \]
  • A feasible allocation is a sequence \(\{c_t, x_t, n_t, k_t\}_{t=0}^\infty\) that satisfies feasibility condition (36.1).

36.3. Equilibrium#

Definition 36.1

A competitive equilibrium with distorting taxes is a budget-feasible government policy, a feasible allocation, and a price system for which, given the price system and the government policy, the allocation solves the household’s problem and the firm’s problem.

36.4. No-arbitrage Condition#

A no-arbitrage argument implies a restriction on prices and tax rates across time.

By rearranging (36.3) and group \(k_t\) at the same \(t\), we can get

(36.6)#\[\begin{split} \begin{aligned} \sum_{t=0}^\infty q_t \left[(1 + \tau_{ct})c_t \right] &\leq \sum_{t=0}^\infty q_t(1 - \tau_{nt})w_t n_t - \sum_{t=0}^\infty q_t \tau_{ht} \\ &+ \sum_{t=1}^\infty\left\{ \left[(1 - \tau_{kt})(\eta_t - \delta) + 1\right]q_t - q_{t-1}\right\}k_t \\ &+ \left[(1 - \tau_{k0})(\eta_0 - \delta) + 1\right]q_0k_0 - \lim_{T \to \infty} q_T k_{T+1} \end{aligned} \end{split}\]

The household inherits a given \(k_0\) that it takes as initial condition and is free to choose \(\{ c_t, n_t, k_{t+1} \}_{t=0}^\infty\).

The household’s budget constraint (36.3) must be bounded in equilibrium due to finite resources.

This imposes a restriction on price and tax sequences.

Specifically, for \(t \geq 1\), the terms multiplying \(k_t\) must equal zero.

If they were strictly positive (negative), the household could arbitrarily increase (decrease) the right-hand side of (36.3) by selecting an arbitrarily large positive (negative) \(k_t\), leading to unbounded profit or arbitrage opportunities:

  • For strictly positive terms, the household could purchase large capital stocks \(k_t\) and profit from their rental services and undepreciated value.

  • For strictly negative terms, the household could engage in “short selling” synthetic units of capital. Both cases would make (36.3) unbounded.

Hence, by setting the terms multiplying \(k_t\) to \(0\) we have the non-arbitrage condition:

(36.7)#\[ \frac{q_t}{q_{t+1}} = \left[(1 - \tau_{kt+1})(\eta_{t+1} - \delta) + 1\right]. \]

Moreover, we have terminal condition:

(36.8)#\[ -\lim_{T \to \infty} q_T k_{T+1} = 0. \]

Zero-profit conditions for the representative firm impose additional restrictions on equilibrium prices and quantities.

The present value of the firm’s profits is

\[ \sum_{t=0}^\infty q_t \left[ F(k_t, n_t) - w_t n_t - \eta_t k_t \right]. \]

Applying Euler’s theorem on linearly homogeneous functions to \(F(k, n)\), the firm’s present value is:

\[ \sum_{t=0}^\infty q_t \left[ (F_{kt} - \eta_t)k_t + (F_{nt} - w_t)n_t \right]. \]

No-arbitrage (or zero-profit) conditions are:

(36.9)#\[ \eta_t = F_{kt}, \quad w_t = F_{nt}. \]

36.5. Household’s First Order Condition#

Household maximize (36.2) under (36.3).

Let \(U_1 = \frac{\partial U}{\partial c}, U_2 = \frac{\partial U}{\partial (1-n)} = -\frac{\partial U}{\partial n}.\), we can derive FOC from the Lagrangian

\[ \mathcal{L} = \sum_{t=0}^\infty \beta^t U(c_t, 1 - n_t) + \mu \left( \sum_{t=0}^\infty q_t \left[(1 + \tau_{ct})c_t - (1 - \tau_{nt})w_t n_t + \ldots \right] \right), \]

First-order necessary conditions for the representative household’s problem are

(36.10)#\[ \frac{\partial \mathcal{L}}{\partial c_t} = \beta^t U_{1}(c_t, 1 - n_t) - \mu q_t (1 + \tau_{ct}) = 0 \]

and

(36.11)#\[ \frac{\partial \mathcal{L}}{\partial n_t} = \beta^t \left(-U_{2t}(c_t, 1 - n_t)\right) - \mu q_t (1 - \tau_{nt}) w_t = 0 \]

Rearranging (36.10) and (36.11), we have

(36.12)#\[ \begin{aligned} \beta^t U_1(c_t, 1 - n_t) = \beta^t U_{1t} = \mu q_t (1 + \tau_{ct}), \end{aligned} \]
(36.13)#\[ \begin{aligned} \beta^t U_2(c_t, 1 - n_t) = \beta^t U_{2t} = \mu q_t (1 - \tau_{nt}) w_t. \end{aligned} \]

Plugging (36.12) into (36.8) and replacing \(q_t\), we get terminal condition

(36.14)#\[ -\lim_{T \to \infty} \beta^T \frac{U_{1T}}{(1 + \tau_{cT})} k_{T+1} = 0. \]

36.6. Computing Equilibria#

To compute an equilibrium, we seek a price system \(\{q_t, \eta_t, w_t\}\), a budget feasible government policy \(\{g_t, \tau_t\} \equiv \{g_t, \tau_{ct}, \tau_{nt}, \tau_{kt}, \tau_{ht}\}\), and an allocation \(\{c_t, n_t, k_{t+1}\}\) that solve a system of nonlinear difference equations consisting of

  • feasibility condition (36.1), no-arbitrage condition for household (36.7) and firms (36.9), household’s first order conditions (36.12) and (36.13).

  • an initial condition \(k_0\) and a terminal condition (36.14).

36.7. Python Code#

We require the following imports

import numpy as np
from scipy.optimize import root
import matplotlib.pyplot as plt
from collections import namedtuple
from mpmath import mp, mpf
from warnings import warn

# Set the precision
mp.dps = 40
mp.pretty = True

We use the mpmath library to perform high-precision arithmetic in the shooting algorithm in cases where the solution diverges due to numerical instability.

We set the following parameters

# Create a namedtuple to store the model parameters
Model = namedtuple("Model", 
            ["β", "γ", "δ", "α", "A"])

def create_model(β=0.95, # discount factor
                 γ=2.0,  # relative risk aversion coefficient
                 δ=0.2,  # depreciation rate
                 α=0.33, # capital share
                 A=1.0   # TFP
                 ):
    """Create a model instance."""
    return Model(β=β, γ=γ, δ=δ, α=α, A=A)

model = create_model()

# Total number of periods
S = 100

36.7.1. Inelastic Labor Supply#

In this lecture, we consider the special case where \(U(c, 1-n) = u(c)\) and \(f(k) := F(k, 1)\).

We rewrite (36.1) with \(f(k) := F(k, 1)\),

(36.15)#\[ k_{t+1} = f(k_t) + (1 - \delta) k_t - g_t - c_t. \]
def next_k(k_t, g_t, c_t, model):
    """
    Capital next period: k_{t+1} = f(k_t) + (1 - δ) * k_t - c_t - g_t
    """
    return f(k_t, model) + (1 - model.δ) * k_t - g_t - c_t

By the properties of a linearly homogeneous production function, we have \(F_k(k, n) = f'(k)\) and \(F_n(k, 1) = f(k, 1) - f'(k)k\).

Substituting (36.12), (36.9), and (36.15) into (36.7), we obtain the Euler equation

(36.16)#\[\begin{split} \begin{aligned} &\frac{u'(f(k_t) + (1 - \delta) k_t - g_t - k_{t+1})}{(1 + \tau_{ct})} \\ &- \beta \frac{u'(f(k_{t+1}) + (1 - \delta) k_{t+1} - g_{t+1} - k_{t+2})}{(1 + \tau_{ct+1})} \\ &\times [(1 - \tau_{kt+1})(f'(k_{t+1}) - \delta) + 1] = 0. \end{aligned} \end{split}\]

This can be simplified to:

(36.17)#\[ \begin{aligned} u'(c_t) = \beta u'(c_{t+1}) \frac{(1 + \tau_{ct})}{(1 + \tau_{ct+1})} [(1 - \tau_{kt+1})(f'(k_{t+1}) - \delta) + 1]. \end{aligned} \]

Equation (36.17) will appear prominently in our equilibrium computation algorithms.

36.7.2. Steady state#

Tax rates and government expenditures act as forcing functions for difference equations (36.15) and (36.17).

Define \(z_t = [g_t, \tau_{kt}, \tau_{ct}]'\).

Represent the second-order difference equation as:

(36.18)#\[ H(k_t, k_{t+1}, k_{t+2}; z_t, z_{t+1}) = 0. \]

We assume that a government policy reaches a steady state such that \(\lim_{t \to \infty} z_t = \bar z\) and that the steady state prevails for \(t > T\).

The terminal steady-state capital stock \(\bar{k}\) satisfies:

\[ H(\bar{k}, \bar{k}, \bar{k}, \bar{z}, \bar{z}) = 0. \]

From difference equation (36.17), we can infer a restriction on the steady-state:

(36.19)#\[\begin{split} \begin{aligned} u'(\bar{c}) &= \beta u'(\bar{c}) \frac{(1 + \bar{\tau}_{c})}{(1 + \bar{\tau}_{c})} [(1 - \bar{\tau}_{k})(f'(\bar{k}) - \delta) + 1]. \\ &\implies 1 = \beta[(1 - \bar{\tau}_{k})(f'(\bar{k}) - \delta) + 1]. \end{aligned} \end{split}\]

36.7.3. Other equilibrium quantities and prices#

Price:

(36.20)#\[ q_t = \frac{\beta^t u'(c_t)}{u'(c_0)} \]
def compute_q_path(c_path, model, S=100):
    """
    Compute q path: q_t = (β^t * u'(c_t)) / u'(c_0)
    """
    q_path = np.zeros_like(c_path)
    for t in range(S):
        q_path[t] = (model.β ** t * 
                     u_prime(c_path[t], model)) / u_prime(c_path[0], model)
    return q_path

Capital rental rate

\[ \eta_t = f'(k_t) \]
def compute_η_path(k_path, model, S=100):
    """
    Compute η path: η_t = f'(k_t)
    """
    η_path = np.zeros_like(k_path)
    for t in range(S):
        η_path[t] = f_prime(k_path[t], model)
    return η_path

Labor rental rate:

\[ w_t = f(k_t) - k_t f'(k_t) \]
def compute_w_path(k_path, η_path, model, S=100):
    """
    Compute w path: w_t = f(k_t) - k_t * f'(k_t)
    """
    A, α = model.A, model.α, model.δ
    w_path = np.zeros_like(k_path)
    for t in range(S):
        w_path[t] = f(k_path[t], model) - k_path[t] * η_path[t]
    return w_path

Gross one-period return on capital:

(36.21)#\[ \bar{R}_{t+1} = \frac{(1 + \tau_{ct})}{(1 + \tau_{ct+1})} \left[(1 - \tau_{kt+1})(f'(k_{t+1}) - \delta) + 1\right] = \frac{(1 + \tau_{ct})}{(1 + \tau_{ct+1})} R_{t, t+1} \]
def compute_R_bar(τ_ct, τ_ctp1, τ_ktp1, k_tp1, model):
    """
    Gross one-period return on capital:
    R̄ = [(1 + τ_c_t) / (1 + τ_c_{t+1})] 
        * { [1 - τ_k_{t+1}] * [f'(k_{t+1}) - δ] + 1 }
    """
    A, α, δ = model.A, model.α, model.δ
    return  ((1 + τ_ct) / (1 + τ_ctp1)) * (
        (1 - τ_ktp1) * (f_prime(k_tp1, model) - δ) + 1) 

def compute_R_bar_path(shocks, k_path, model, S=100):
    """
    Compute R̄ path over time.
    """
    A, α, δ = model.A, model.α, model.δ
    R_bar_path = np.zeros(S + 1)
    for t in range(S):
        R_bar_path[t] = compute_R_bar(
            shocks['τ_c'][t], shocks['τ_c'][t + 1], shocks['τ_k'][t + 1],
            k_path[t + 1], model)
    R_bar_path[S] = R_bar_path[S - 1]
    return R_bar_path

One-period discount factor:

(36.22)#\[ R^{-1}_{t, t+1} = \frac{q_{t+1}}{q_{t}} = m_{t, t+1} = \beta \frac{u'(c_{t+1})}{u'(c_t)} \frac{(1 + \tau_{ct})}{(1 + \tau_{ct+1})} \]

Net one-period rate of interest:

(36.23)#\[ r_{t, t+1} \equiv R_{t, t+1} - 1 = (1 - \tau_{k, t+1})(f'(k_{t+1}) - \delta) \]

By (36.22) and \(r_{t, t+1} = - \ln(\frac{q_{t+1}}{q_t})\), we have

\[ R_{t, t+s} = e^{s \cdot r_{t, t+s}}. \]

Then by (36.23), we have

\[ \frac{q_{t+s}}{q_t} = e^{-s \cdot r_{t, t+s}}. \]

Rearranging the above equation, we have

\[ r_{t, t+s} = -\frac{1}{s} \ln\left(\frac{q_{t+s}}{q_t}\right). \]
def compute_rts_path(q_path, S, t):
    """
    Compute r path:
    r_t,t+s = - (1/s) * ln(q_{t+s} / q_t)
    """
    s = np.arange(1, S + 1) 
    q_path = np.array([float(q) for q in q_path]) 
    
    with np.errstate(divide='ignore', invalid='ignore'):
        rts_path = - np.log(q_path[t + s] / q_path[t]) / s
    return rts_path

36.8. Some functional forms#

We assume that the representative household’ period utility has the following CRRA (constant relative risk aversion) form

\[ u(c) = \frac{c^{1 - \gamma}}{1 - \gamma} \]
def u_prime(c, model):
    """
    Marginal utility: u'(c) = c^{-γ}
    """
    return c ** (-model.γ)

By substituting (36.21) into (36.17), we obtain

(36.24)#\[ c_{t+1} = c_t \left[ \beta \frac{(1 + \tau_{ct})}{(1 + \tau_{ct+1})} \left[(1 - \tau_{k, t+1})(f'(k_{t+1}) - \delta) + 1 \right] \right]^{\frac{1}{\gamma}} = c_t \left[ \beta \overline{R}_{t+1} \right]^{\frac{1}{\gamma}} \]
def next_c(c_t, R_bar, model):
    """
    Consumption next period: c_{t+1} = c_t * (β * R̄)^{1/γ}
    """
    β, γ = model.β, model.γ
    return c_t * (β * R_bar) ** (1 / γ)

For the production function we assume a Cobb-Douglas form:

\[ F(k, 1) = A k^\alpha \]
def f(k, model): 
    """
    Production function: f(k) = A * k^{α}
    """
    A, α = model.A, model.α
    return A * k ** α

def f_prime(k, model):
    """
    Marginal product of capital: f'(k) = α * A * k^{α - 1}
    """
    A, α = model.A, model.α
    return α * A * k ** (α - 1)

36.9. Computation#

We describe two ways to compute an equilibrium:

  • a shooting algorithm

  • a residual-minimization method that focuses on imposing Euler equation (36.17) and the feasibility condition (36.15).

36.9.1. Shooting Algorithm#

This algorithm deploys the following steps.

  1. Solve the equation (36.19) for the terminal steady-state capital \(\bar{k}\) that corresponds to the permanent policy vector \(\bar{z}\).

  2. Select a large time index \(S \gg T\), guess an initial consumption rate \(c_0\), and use the equation (36.15) to solve for \(k_1\).

  3. Use the equation (36.24) to determine \(c_{t+1}\). Then, apply the equation (36.15) to compute \(k_{t+2}\).

  4. Iterate step 3 to compute candidate values \(\hat{k}_t\) for \(t = 1, \dots, S\).

  5. Compute the difference \(\hat{k}_S - \bar{k}\). If \(\left| \hat{k}_S - \bar{k} \right| > \epsilon\) for some small \(\epsilon\), adjust \(c_0\) and repeat steps 2-5.

  6. Adjust \(c_0\) iteratively using the bisection method to find a value that ensures \(\left| \hat{k}_S - \bar{k} \right| < \epsilon\).

The following code implements these steps.

# Steady-state calculation
def steady_states(model, g_ss, τ_k_ss=0.0):
    """
    Steady-state values:
    - Capital: (1 - τ_k_ss) * [α * A * k_ss^{α - 1} - δ] = (1 / β) - 1
    - Consumption: c_ss = A * k_ss^{α} - δ * k_ss - g_ss
    """
    β, δ, α, A = model.β, model.δ, model.α, model.A
    numerator = δ + (1 / β - 1) / (1 - τ_k_ss)
    denominator = α * A
    k_ss = (numerator / denominator) ** (1 / (α - 1))
    c_ss = A * k_ss ** α - δ * k_ss - g_ss
    return k_ss, c_ss

def shooting_algorithm(c0, k0, shocks, S, model):
    """
    Shooting algorithm for given initial c0 and k0.
    """
    # Convert shocks to high-precision
    g_path, τ_c_path, τ_k_path = (
        list(map(mpf, shocks[key])) for key in ['g', 'τ_c', 'τ_k']
    )

    # Initialize paths with initial values
    c_path = [mpf(c0)] + [mpf(0)] * S
    k_path = [mpf(k0)] + [mpf(0)] * S

    # Generate paths for k_t and c_t
    for t in range(S):
        k_t, c_t, g_t = k_path[t], c_path[t], g_path[t]

        # Calculate next period's capital
        k_tp1 = next_k(k_t, g_t, c_t, model)
        
        # Failure due to negative capital
        if k_tp1 < mpf(0):
            return None, None 
        k_path[t + 1] = k_tp1

        # Calculate next period's consumption
        R_bar = compute_R_bar(τ_c_path[t], τ_c_path[t + 1], 
                              τ_k_path[t + 1], k_tp1, model)
        c_tp1 = next_c(c_t, R_bar, model)

        # Failure due to negative consumption
        if c_tp1 < mpf(0):
            return None, None
        c_path[t + 1] = c_tp1

    return k_path, c_path


def bisection_c0(c0_guess, k0, shocks, S, model, 
                 tol=mpf('1e-6'), max_iter=1000, verbose=False):
    """
    Bisection method to find optimal initial consumption c0.
    """
    k_ss_final, _ = steady_states(model, 
                                  mpf(shocks['g'][-1]), 
                                  mpf(shocks['τ_k'][-1]))
    c0_lower, c0_upper = mpf(0), f(k_ss_final, model)

    c0 = c0_guess
    for iter_count in range(max_iter):
        k_path, _ = shooting_algorithm(c0, k0, shocks, S, model)
        
        # Adjust upper bound when shooting fails
        if k_path is None:
            if verbose:
                print(f"Iteration {iter_count + 1}: shooting failed with c0 = {c0}")
            c0_upper = c0
        else:
            error = k_path[-1] - k_ss_final
            if verbose and iter_count % 100 == 0:
                print(f"Iteration {iter_count + 1}: c0 = {c0}, error = {error}")

            # Check for convergence
            if abs(error) < tol:
                print(f"Converged successfully on iteration {iter_count + 1}")
                return c0 

            # Update bounds based on the error
            if error > mpf(0):
                c0_lower = c0
            else:
                c0_upper = c0

        # Calculate the new midpoint for bisection
        c0 = (c0_lower + c0_upper) / mpf('2')

    # Return the last computed c0 if convergence was not achieved
    # Send a Warning message when this happens
    warn(f"Converged failed. Returning the last c0 = {c0}", stacklevel=2)
    return c0

def run_shooting(shocks, S, model, c0_func=bisection_c0, shooting_func=shooting_algorithm):
    """
    Runs the shooting algorithm.
    """
    # Compute initial steady states
    k0, c0 = steady_states(model, mpf(shocks['g'][0]), mpf(shocks['τ_k'][0]))
    
    # Find the optimal initial consumption
    optimal_c0 = c0_func(c0, k0, shocks, S, model)
    print(f"Parameters: {model}")
    print(f"Optimal initial consumption c0: {mp.nstr(optimal_c0, 7)} \n")
    
    # Simulate the model
    k_path, c_path = shooting_func(optimal_c0, k0, shocks, S, model)
    
    # Combine and return the results
    return np.column_stack([k_path, c_path])

36.9.2. Experiments#

Let’s run some experiments.

  1. A foreseen once-and-for-all increase in \(g\) from 0.2 to 0.4 occurring in period 10,

  2. A foreseen once-and-for-all increase in \(\tau_c\) from 0.0 to 0.2 occurring in period 10,

  3. A foreseen once-and-for-all increase in \(\tau_k\) from 0.0 to 0.2 occurring in period 10, and

  4. A foreseen one-time increase in \(g\) from 0.2 to 0.4 in period 10, after which \(g\) reverts to 0.2 permanently.

To start, we prepare sequences that we’ll used to initialize our iterative algorithm.

We will start from an initial steady state and apply shocks at an the indicated time.

def plot_results(solution, k_ss, c_ss, shocks, shock_param, 
                 axes, model, label='', linestyle='-', T=40):
    """
    Plot the results of the simulation replicating graphs in RMT.
    """
    k_path = solution[:, 0]
    c_path = solution[:, 1]

    axes[0].plot(k_path[:T], linestyle=linestyle, label=label)
    axes[0].axhline(k_ss, linestyle='--', color='black')
    axes[0].set_title('k')

    # Plot for c
    axes[1].plot(c_path[:T], linestyle=linestyle, label=label)
    axes[1].axhline(c_ss, linestyle='--', color='black')
    axes[1].set_title('c')

    # Plot for g
    R_bar_path = compute_R_bar_path(shocks, k_path, model, S)

    axes[2].plot(R_bar_path[:T], linestyle=linestyle, label=label)
    axes[2].set_title('$\overline{R}$')
    axes[2].axhline(1 / model.β, linestyle='--', color='black')
    
    η_path = compute_η_path(k_path, model, S=T)
    η_ss = model.α * model.A * k_ss ** (model.α - 1)
    
    axes[3].plot(η_path[:T], linestyle=linestyle, label=label)
    axes[3].axhline(η_ss, linestyle='--', color='black')
    axes[3].set_title(r'$\eta$')
    
    axes[4].plot(shocks[shock_param][:T], linestyle=linestyle, label=label)
    axes[4].axhline(shocks[shock_param][0], linestyle='--', color='black')
    axes[4].set_title(rf'${shock_param}$')
<>:22: SyntaxWarning: invalid escape sequence '\o'
<>:22: SyntaxWarning: invalid escape sequence '\o'
/tmp/ipykernel_181/1339523311.py:22: SyntaxWarning: invalid escape sequence '\o'
  axes[2].set_title('$\overline{R}$')

Experiment 1: Foreseen once-and-for-all increase in \(g\) from 0.2 to 0.4 in period 10

The figure below shows consequences of a foreseen permanent increase in \(g\) at \(t = T = 10\) that is financed by an increase in lump-sum taxes

# Define shocks as a dictionary
shocks = {
    'g': np.concatenate((np.repeat(0.2, 10), np.repeat(0.4, S - 9))),
    'τ_c': np.repeat(0.0, S + 1),
    'τ_k': np.repeat(0.0, S + 1)
}

k_ss_initial, c_ss_initial = steady_states(model, 
                                           shocks['g'][0], 
                                           shocks['τ_k'][0])

print(f"Steady-state capital: {k_ss_initial:.4f}")
print(f"Steady-state consumption: {c_ss_initial:.4f}")

solution = run_shooting(shocks, S, model)

fig, axes = plt.subplots(2, 3, figsize=(10, 8))
axes = axes.flatten()

plot_results(solution, k_ss_initial, 
             c_ss_initial, shocks, 'g', axes, model, T=40)

for ax in axes[5:]:
    fig.delaxes(ax)

plt.tight_layout()
plt.show()
Steady-state capital: 1.4900
Steady-state consumption: 0.6426
Converged successfully on iteration 44
Parameters: Model(β=0.95, γ=2.0, δ=0.2, α=0.33, A=1.0)
Optimal initial consumption c0: 0.609242 
_images/05066dd2f030bd7d1837ea9617d4adcf3f775ad2dcbf1a69f21ed26259fa8618.png

The above figures indicate that an equilibrium consumption smoothing mechanism is at work, driven by the representative consumer’s preference for smooth consumption paths coming from the curvature of its one-period utility function.

  • The steady-state value of the capital stock remains unaffected:

    • This follows from the fact that \(g\) disappears from the steady state version of the Euler equation ((36.19)).

  • Consumption begins to decline gradually before time \(T\) due to increased government consumption:

    • Households reduce consumption to offset government spending, which is financed through increased lump-sum taxes.

    • The competitive economy signals households to consume less through an increase in the stream of lump-sum taxes.

    • Households, caring about the present value rather than the timing of taxes, experience an adverse wealth effect on consumption, leading to an immediate response.

  • Capital gradually accumulates between time \(0\) and \(T\) due to increased savings and reduces gradually after time \(T\):

    • This temporal variation in capital stock smooths consumption over time, driven by the representative consumer’s consumption-smoothing motive.

Let’s collect the procedures used above into a function that runs the solver and draws plots for a given experiment

Hide code cell source
def experiment_model(shocks, S, model, solver, plot_func, policy_shock, T=40):
    """
    Run the shooting algorithm given a model and plot the results.
    """
    k0, c0 = steady_states(model, shocks['g'][0], shocks['τ_k'][0])
    
    print(f"Steady-state capital: {k0:.4f}")
    print(f"Steady-state consumption: {c0:.4f}")
    print('-'*64)
    
    fig, axes = plt.subplots(2, 3, figsize=(10, 8))
    axes = axes.flatten()

    solution = solver(shocks, S, model)
    plot_func(solution, k0, c0, 
              shocks, policy_shock, axes, model, T=T)

    for ax in axes[5:]:
        fig.delaxes(ax)

    plt.tight_layout()
    plt.show()

The following figure compares responses to a foreseen increase in \(g\) at \(t = 10\) for two economies:

  • our original economy with \(\gamma = 2\), shown in the solid line, and

  • an otherwise identical economy with \(\gamma = 0.2\).

This comparison interest us because the utility curvature parameter \(\gamma\) governs the household’s willingness to substitute consumption over time, and thus it preferences about smoothness of consumption paths over time.

# Solve the model using shooting
solution = run_shooting(shocks, S, model)

# Compute the initial steady states
k_ss_initial, c_ss_initial = steady_states(model, 
                                           shocks['g'][0], 
                                           shocks['τ_k'][0])

# Plot the solution for γ=2
fig, axes = plt.subplots(2, 3, figsize=(10, 8))
axes = axes.flatten()

label = fr"$\gamma = {model.γ}$"
plot_results(solution, k_ss_initial, c_ss_initial, 
             shocks, 'g', axes, model, label=label, 
             T=40)

# Solve and plot the result for γ=0.2
model_γ2 = create_model(γ=0.2)
solution = run_shooting(shocks, S, model_γ2)

plot_results(solution, k_ss_initial, c_ss_initial, 
             shocks, 'g', axes, model_γ2, 
             label=fr"$\gamma = {model_γ2.γ}$", 
             linestyle='-.', T=40)

handles, labels = axes[0].get_legend_handles_labels()  
fig.legend(handles, labels, loc='lower right', 
           ncol=3, fontsize=14, bbox_to_anchor=(1, 0.1))  

for ax in axes[5:]:
    fig.delaxes(ax)
    
plt.tight_layout()
plt.show()
Converged successfully on iteration 44
Parameters: Model(β=0.95, γ=2.0, δ=0.2, α=0.33, A=1.0)
Optimal initial consumption c0: 0.609242 
Converged successfully on iteration 91
Parameters: Model(β=0.95, γ=0.2, δ=0.2, α=0.33, A=1.0)
Optimal initial consumption c0: 0.642033 
_images/2c0e5bffd0685364a02e458b86765cecd0b4780c1d090f437aa86e142228e19c.png

The outcomes indicate that lowering \(\gamma\) affects both the consumption and capital stock paths because - it increases the representative consumer’s willingness to substitute consumption across time:

  • Consumption path:

    • When \(\gamma = 0.2\), consumption becomes less smooth compared to \(\gamma = 2\).

    • For \(\gamma = 0.2\), consumption mirrors the government expenditure path more closely, staying higher until \(t = 10\).

  • Capital stock path:

    • With \(\gamma = 0.2\), there are smaller build-ups and drawdowns of capital stock.

    • There are also smaller fluctuations in \(\bar{R}\) and \(\eta\).

Let’s write another function that runs the solver and draws plots for these two experiments

Hide code cell source
def experiment_two_models(shocks, S, model_1, model_2, solver, plot_func, 
                          policy_shock, legend_label_fun=None, T=40):
    """
    Compares and plots results of the shooting algorithm for two models.
    """
    k0, c0 = steady_states(model, shocks['g'][0], shocks['τ_k'][0])
    print(f"Steady-state capital: {k0:.4f}")
    print(f"Steady-state consumption: {c0:.4f}")
    print('-'*64)
    
    # Use a default legend labeling function if none is provided
    if legend_label_fun is None:
        legend_label_fun = lambda model: fr"$\gamma = {model.γ}$"

    # Set up the figure and axes
    fig, axes = plt.subplots(2, 3, figsize=(10, 8))
    axes = axes.flatten()

    # Function to run and plot for each model
    def run_and_plot(model, linestyle='-'):
        solution = solver(shocks, S, model)
        plot_func(solution, k0, c0, shocks, policy_shock, axes, model, 
                  label=legend_label_fun(model), linestyle=linestyle, T=T)

    # Plot for both models
    run_and_plot(model_1)
    run_and_plot(model_2, linestyle='-.')

    # Set legend using labels from the first axis
    handles, labels = axes[0].get_legend_handles_labels()
    fig.legend(handles, labels, loc='lower right', ncol=3, 
               fontsize=14, bbox_to_anchor=(1, 0.1))

    # Remove extra axes and tidy up the layout
    for ax in axes[5:]:
        fig.delaxes(ax)
    
    plt.tight_layout()
    plt.show()

Now we plot other equilibrium quantities:

def plot_prices(solution, c_ss, shock_param, axes,
                model, label='', linestyle='-', T=40):
    """
    Compares and plots prices
    """
    α, β, δ, γ, A = model.α, model.β, model.δ, model.γ, model.A
    
    k_path = solution[:, 0]
    c_path = solution[:, 1]

    # Plot for c
    axes[0].plot(c_path[:T], linestyle=linestyle, label=label)
    axes[0].axhline(c_ss, linestyle='--', color='black')
    axes[0].set_title('c')
    
    # Plot for q
    q_path = compute_q_path(c_path, model, S=S)
    axes[1].plot(q_path[:T], linestyle=linestyle, label=label)
    axes[1].plot(β**np.arange(T), linestyle='--', color='black')
    axes[1].set_title('q')
    
    # Plot for r_{t,t+1}
    R_bar_path = compute_R_bar_path(shocks, k_path, model, S)
    axes[2].plot(R_bar_path[:T] - 1, linestyle=linestyle, label=label)
    axes[2].axhline(1 / β - 1, linestyle='--', color='black')
    axes[2].set_title('$r_{t,t+1}$')

    # Plot for r_{t,t+s}
    for style, s in zip(['-', '-.', '--'], [0, 10, 60]):
        rts_path = compute_rts_path(q_path, T, s)
        axes[3].plot(rts_path, linestyle=style, 
                     color='black' if style == '--' else None,
                     label=f'$t={s}$')
        axes[3].set_xlabel('s')
        axes[3].set_title('$r_{t,t+s}$')

    # Plot for g
    axes[4].plot(shocks[shock_param][:T], linestyle=linestyle, label=label)
    axes[4].axhline(shocks[shock_param][0], linestyle='--', color='black')
    axes[4].set_title(shock_param)

For \(\gamma = 2\) again, the next figure describes the response of \(q_t\) and the term structure of interest rates to a foreseen increase in \(g_t\) at \(t = 10\)

solution = run_shooting(shocks, S, model)

fig, axes = plt.subplots(2, 3, figsize=(10, 8))
axes = axes.flatten()

plot_prices(solution, c_ss_initial, 'g', axes, model, T=40)

for ax in axes[5:]:
    fig.delaxes(ax)

handles, labels = axes[3].get_legend_handles_labels()  
fig.legend(handles, labels, title=r"$r_{t,t+s}$ with ", loc='lower right', ncol=3, fontsize=10, bbox_to_anchor=(1, 0.1))  
plt.tight_layout()
plt.show()
Converged successfully on iteration 44
Parameters: Model(β=0.95, γ=2.0, δ=0.2, α=0.33, A=1.0)
Optimal initial consumption c0: 0.609242 
_images/4e3a32649d2f429dfa9cf21a3f541b18bb4dcaa878ff761e4beaed298bda96f6.png

The second panel on the top compares \(q_t\) for the initial steady state with \(q_t\) after the increase in \(g\) is foreseen at \(t = 0\), while the third panel compares the implied short rate \(r_t\).

The fourth panel shows the term structure of interest rates at \(t=0\), \(t=10\), and \(t=60\).

Notice that, by \(t = 60\), the system has converged to the new steady state, and the term structure of interest rates becomes flat.

At \(t = 10\), the term structure of interest rates is upward sloping.

This upward slope reflects the expected increase in the rate of growth of consumption over time, as shown in the consumption panel.

At \(t = 0\), the term structure of interest rates exhibits a “U-shaped” pattern:

  • It declines until maturity at \(s = 10\).

  • After \(s = 10\), it increases for longer maturities.

This pattern aligns with the pattern of consumption growth in the first two figures, which declines at an increasing rate until \(t = 10\) and then declines at a decreasing rate afterward.

Experiment 2: Foreseen once-and-for-all increase in \(\tau_c\) from 0.0 to 0.2 in period 10

With an inelastic labor supply, the Euler equation (36.16) and the other equilibrium conditions show that

  • constant consumption taxes do not distort decisions, but

  • anticipated changes in them do.

Indeed, (36.16) or (36.17) indicates that a foreseen in- crease in \(\tau_{ct}\) (i.e., a decrease in \((1+\tau_{ct})\) \((1+\tau_{ct+1})\)) operates like an increase in \(\tau_{kt}\).

The following figure portrays the response to a foreseen increase in the consumption tax \(\tau_c\).

shocks = {
    'g': np.repeat(0.2, S + 1),
    'τ_c': np.concatenate((np.repeat(0.0, 10), np.repeat(0.2, S - 9))),
    'τ_k': np.repeat(0.0, S + 1)
}

experiment_model(shocks, S, model, run_shooting, plot_results, 'τ_c')
Steady-state capital: 1.4900
Steady-state consumption: 0.6426
----------------------------------------------------------------
Converged successfully on iteration 51
Parameters: Model(β=0.95, γ=2.0, δ=0.2, α=0.33, A=1.0)
Optimal initial consumption c0: 0.6492796 
_images/79b659074e16d149eb700337defa15cc620f2bea88d99a677b0fe30254fb08aa.png

Evidently all variables in the figures above eventually return to their initial steady-state values.

The anticipated increase in \(\tau_{ct}\) leads to variations in consumption and capital stock across time:

  • At \(t = 0\):

    • Anticipation of the increase in \(\tau_c\) causes an immediate jump in consumption.

    • This is followed by a consumption binge that sends the capital stock downward until \(t = T = 10\).

  • Between \(t = 0\) and \(t = T = 10\):

    • The decline in the capital stock raises \(\bar{R}\) over time.

    • The equilibrium conditions require the growth rate of consumption to rise until \(t = T\).

  • At \(t = T = 10\):

    • The jump in \(\tau_c\) depresses \(\bar{R}\) below \(1\), causing a sharp drop in consumption.

  • After \(T = 10\):

    • The effects of anticipated distortion are over, and the economy gradually adjusts to the lower capital stock.

    • Capital must now rise, requiring austerity —consumption plummets after \(t = T\), indicated by lower levels of consumption.

    • The interest rate gradually declines, and consumption grows at a diminishing rate along the path to the terminal steady-state.

Experiment 3: Foreseen once-and-for-all increase in \(\tau_k\) from 0.0 to 0.2 in period 10

For the two \(\gamma\) values 2 and 0.2, the next figure shows the response to a foreseen permanent jump in \(\tau_{kt}\) at \(t = T = 10\).

shocks = {
    'g': np.repeat(0.2, S + 1),
    'τ_c': np.repeat(0.0, S + 1),
    'τ_k': np.concatenate((np.repeat(0.0, 10), np.repeat(0.2, S - 9))) 
}

experiment_two_models(shocks, S, model, model_γ2, 
                run_shooting, plot_results, 'τ_k')
Steady-state capital: 1.4900
Steady-state consumption: 0.6426
----------------------------------------------------------------
Converged successfully on iteration 50
Parameters: Model(β=0.95, γ=2.0, δ=0.2, α=0.33, A=1.0)
Optimal initial consumption c0: 0.6448856 
Converged successfully on iteration 101
Parameters: Model(β=0.95, γ=0.2, δ=0.2, α=0.33, A=1.0)
Optimal initial consumption c0: 0.6428408 
_images/3dd6a423fd7a04ebf4613953f41b10cac4aae0118e85810d44e17ea6fd6bb8a3.png

The path of government expenditures remains fixed

  • the increase in \(\tau_{kt}\) is offset by a reduction in the present value of lump-sum taxes to keep the budget balanced.

The figure shows that:

  • Anticipation of the increase in \(\tau_{kt}\) leads to immediate decline in capital stock due to increased current consumption and a growing consumption flow.

  • \(\bar{R}\) starts rising at \(t = 0\) and peaks at \(t = 9\), and at \(t = 10\), \(\bar{R}\) drops sharply due to the tax change.

    • Variations in \(\bar{R}\) align with the impact of the tax increase at \(t = 10\) on consumption across time.

  • Transition dynamics push \(k_t\) (capital stock) toward a new, lower steady-state level. In the new steady state:

    • Consumption is lower due to reduced output from the lower capital stock.

    • Smoother consumption paths occur when \(\gamma = 2\) than when \(\gamma = 0.2\).

So far we have explored consequences of foreseen once-and-for-all changes in government policy. Next we describe some experiments in which there is a foreseen one-time change in a policy variable (a “pulse”).

Experiment 4: Foreseen one-time increase in \(g\) from 0.2 to 0.4 in period 10, after which \(g\) returns to 0.2 forever

g_path = np.repeat(0.2, S + 1)
g_path[10] = 0.4

shocks = {
    'g': g_path,
    'τ_c': np.repeat(0.0, S + 1),
    'τ_k': np.repeat(0.0, S + 1)
}

experiment_model(shocks, S, model, run_shooting, plot_results, 'g')
Steady-state capital: 1.4900
Steady-state consumption: 0.6426
----------------------------------------------------------------
Converged successfully on iteration 51
Parameters: Model(β=0.95, γ=2.0, δ=0.2, α=0.33, A=1.0)
Optimal initial consumption c0: 0.6378298 
_images/2b73cde9f5746da00826a3d196cf7f8f866ec3d6e118f2412a2e734e8703522f.png

The figure indicates how:

  • Consumption:

    • Drops immediately upon announcement of the policy and continues to decline over time in anticipation of the one-time surge in \(g\).

    • After the shock at \(t = 10\), consumption begins to recover, rising at a diminishing rate toward its steady-state value.

  • Capital and \(\bar{R}\):

    • Before \(t = 10\), capital accumulates as interest rate changes induce households to prepare for the anticipated increase in government spending.

    • At \(t = 10\), the capital stock sharply decreases as the government consumes part of it.

    • \(\bar{R}\) jumps above its steady-state value due to the capital reduction and then gradually declines toward its steady-state level.

36.9.3. Method 2: Residual Minimization#

The second method involves minimizing residuals (i.e., deviations from equalities) of the following equations:

  • The Euler equation (36.17):

    \[ 1 = \beta \left(\frac{c_{t+1}}{c_t}\right)^{-\gamma} \frac{(1+\tau_{ct})}{(1+\tau_{ct+1})} \left[(1 - \tau_{kt+1})(\alpha A k_{t+1}^{\alpha-1} - \delta) + 1 \right] \]
  • The feasibility condition (36.15):

    \[ k_{t+1} = A k_{t}^{\alpha} + (1 - \delta) k_t - g_t - c_t. \]
# Euler's equation and feasibility condition 
def euler_residual(c_t, c_tp1, τ_c_t, τ_c_tp1, τ_k_tp1, k_tp1, model):
    """
    Computes the residuals for Euler's equation.
    """
    β, γ, δ, α, A = model.β, model.γ, model.δ, model.α, model.A
    η_tp1 = α * A * k_tp1 ** (α - 1)
    return β * (c_tp1 / c_t) ** (-γ) * (1 + τ_c_t) / (1 + τ_c_tp1) * (
        (1 - τ_k_tp1) * (η_tp1 - δ) + 1) - 1

def feasi_residual(k_t, k_tm1, c_tm1, g_t, model):
    """
    Computes the residuals for feasibility condition.
    """
    α, A, δ = model.α, model.A, model.δ
    return k_t - (A * k_tm1 ** α + (1 - δ) * k_tm1 - c_tm1 - g_t)

The algorithm proceeds follows:

  1. Find initial steady state \(k_0\) based on the government plan at \(t=0\).

  2. Initialize a sequence of initial guesses \(\{\hat{c}_t, \hat{k}_t\}_{t=0}^{S}\).

  3. Compute residuals \(l_a\) and \(l_k\) for \(t = 0, \dots, S\), as well as \(l_{k_0}\) for \(t = 0\) and \(l_{k_S}\) for \(t = S\):

    • Compute the Euler equation residual for \(t = 0, \dots, S\) using (36.17):

      \[ l_{ta} = \beta u'(c_{t+1}) \frac{(1 + \tau_{ct})}{(1 + \tau_{ct+1})} \left[(1 - \tau_{kt+1})(f'(k_{t+1}) - \delta) + 1 \right] - 1 \]
    • Compute the feasibility condition residual for \(t = 1, \dots, S-1\) using (36.15):

      \[ l_{tk} = k_{t+1} - f(k_t) - (1 - \delta)k_t + g_t + c_t \]
    • Compute the residual for the initial condition for \(k_0\) using (36.19) and the initial capital \(k_0\):

      \[ l_{k_0} = 1 - \beta \left[ (1 - \tau_{k0}) \left(f'(k_0) - \delta \right) + 1 \right] \]
    • Compute the residual for the terminal condition for \(t = S\) using (36.17) under the assumptions \(c_t = c_{t+1} = c_S\), \(k_t = k_{t+1} = k_S\), \(\tau_{ct} = \tau_{ct+1} = \tau_{cS}\), and \(\tau_{kt} = \tau_{kt+1} = \tau_{kS}\):

      \[ l_{k_S} = \beta u'(c_S) \frac{(1 + \tau_{cS})}{(1 + \tau_{cS})} \left[(1 - \tau_{kS})(f'(k_S) - \delta) + 1 \right] - 1 \]
  4. Iteratively adjust guesses for \(\{\hat{c}_t, \hat{k}_t\}_{t=0}^{S}\) to minimize residuals \(l_{k_0}\), \(l_{ta}\), \(l_{tk}\), and \(l_{k_S}\) for \(t = 0, \dots, S\).

# Computing residuals as objective function to minimize
def compute_residuals(vars_flat, k_init, S, shocks, model):
    """
    Compute a vector of residuals under Euler's equation, feasibility condition, 
    and boundary conditions.
    """
    k, c = vars_flat.reshape((S + 1, 2)).T
    residuals = np.zeros(2 * S + 2)

    # Initial condition for capital
    residuals[0] = k[0] - k_init

    # Compute residuals for each time step
    for t in range(S):
        residuals[2 * t + 1] = euler_residual(
            c[t], c[t + 1],
            shocks['τ_c'][t], shocks['τ_c'][t + 1], shocks['τ_k'][t + 1],
            k[t + 1], model
        )
        residuals[2 * t + 2] = feasi_residual(
            k[t + 1], k[t], c[t],
            shocks['g'][t], model
        )

    # Terminal condition
    residuals[-1] = euler_residual(
        c[S], c[S],
        shocks['τ_c'][S], shocks['τ_c'][S], shocks['τ_k'][S],
        k[S], model
    )
    
    return residuals

# Root-finding Algorithm to minimize the residual
def run_min(shocks, S, model):
    """
    Root-finding algorithm to minimize the vector of residuals.
    """
    k_ss, c_ss = steady_states(model, shocks['g'][0], shocks['τ_k'][0])
    
    # Initial guess for the solution path
    initial_guess = np.column_stack(
        (np.full(S + 1, k_ss), np.full(S + 1, c_ss))).flatten()

    # Solve the system using root-finding
    sol = root(compute_residuals, initial_guess, 
               args=(k_ss, S, shocks, model), tol=1e-8)

    # Reshape solution to get time paths for k and c
    return sol.x.reshape((S + 1, 2))

We found that method 2 did not encounter numerical stability issues, so using mp.mpf is not necessary.

We leave as exercises replicating some of our experiments by using the second method.

36.10. Exercises#

Exercise 36.1

Replicate the plots of our four experiments using the second method of residual minimization:

  1. A foreseen once-and-for-all increase in \(g\) from 0.2 to 0.4 occurring in period 10,

  2. A foreseen once-and-for-all increase in \(\tau_c\) from 0.0 to 0.2 occurring in period 10,

  3. A foreseen once-and-for-all increase in \(\tau_k\) from 0.0 to 0.2 occurring in period 10, and

  4. A foreseen one-time increase in \(g\) from 0.2 to 0.4 in period 10, after which \(g\) reverts to 0.2 permanently,

Exercise 36.2

Design a new experiment where the government expenditure \(g\) increases from \(0.2\) to \(0.4\) in period \(10\), and then decreases to \(0.1\) in period \(20\) permanently.