diff --git a/code/estimark/chris_parameters.py b/code/estimark/chris_parameters.py new file mode 100644 index 0000000..4a2764c --- /dev/null +++ b/code/estimark/chris_parameters.py @@ -0,0 +1,226 @@ +"""Specifies the full set of calibrated values required to estimate the EstimatingMicroDSOPs +model. The empirical data is stored in a separate csv file and is loaded in setup_scf_data. +""" + +import numpy as np +from HARK.Calibration.Income.IncomeTools import CGM_income, parse_income_spec +from HARK.ConsumptionSaving.ConsIndShockModel import init_lifecycle +from HARK.datasets.life_tables.us_ssa.SSATools import parse_ssa_life_table +from HARK.distribution import DiscreteDistribution + +# --------------------------------------------------------------------------------- +# - Define all of the model parameters for EstimatingMicroDSOPs and ConsumerExamples - +# --------------------------------------------------------------------------------- + +# Assets grid +exp_nest = 1 # Number of times to "exponentially nest" when constructing a_grid +aXtraMin = 0.001 # Minimum end-of-period "assets above minimum" value +aXtraMax = 100 # Maximum end-of-period "assets above minimum" value +aXtraCount = 20 # Number of points in the grid of "assets above minimum" + +# Artificial borrowing constraint +BoroCnstArt = 0.0 # imposed minimum level of end-of period assets +Rfree = 1.03 # Interest factor on assets + +# Use cubic spline interpolation when True, linear interpolation when False +CubicBool = False +vFuncBool = False # Whether to calculate the value function during solution + +# Income process parameters +# Number of points in discrete approximation to permanent income shocks +PermShkCount = 7 +# Number of points in discrete approximation to transitory income shocks +TranShkCount = 7 +UnempPrb = 0.05 # Probability of unemployment while working +UnempPrbRet = 0.005 # Probability of "unemployment" while retired # maybe one more zero +IncUnemp = 0.3 # Unemployment benefits replacement rate +IncUnempRet = 0.0 # "Unemployment" benefits when retired +ss_variances = True # Use the Sabelhaus-Song variance profiles +education = "College" # Education level for income process + +# Population age parameters +final_age = 120 # Age at which the problem ends (die with certainty) +retirement_age = 125 # Age at which the consumer retires +initial_age = 25 # Age at which the consumer enters the model +final_age_data = 95 # Age at which the data ends +age_interval = 5 # Interval between age groups + +# Three point discrete distribution of initial w +init_w_to_y = np.array([0.17, 0.5, 0.83]) +# Equiprobable discrete distribution of initial w +prob_w_to_y = np.array([0.33333, 0.33333, 0.33334]) +num_agents = 10000 # Number of agents to simulate + +# Bootstrap options +bootstrap_size = 50 # Number of re-estimations to do during bootstrap +seed = 1132023 # Just an integer to seed the estimation + + +params_to_estimate = ["CRRA"] +# Initial guess of the coefficient of relative risk aversion during estimation (rho) +init_CRRA = 5.0 +# Initial guess of the adjustment to the discount factor during estimation (beth) +init_DiscFac = 1.0 +# Bounds for beth; if violated, objective function returns "penalty value" +bounds_DiscFac = [0.5, 1.1] +# Bounds for rho; if violated, objective function returns "penalty value" +bounds_CRRA = [1.1, 20.0] + +init_WealthShare = 0.5 # Initial guess of the wealth share parameter +bounds_WealthShare = [0.0, 1.0] # Bounds for the wealth share parameter + + +###################################################################### +# Constructed parameters +###################################################################### + +# Total number of periods in the model +terminal_t = final_age - initial_age +retirement_t = retirement_age - initial_age - 1 + +# Income +income_spec = CGM_income[education] +# Replace retirement age +income_spec["age_ret"] = retirement_age +inc_calib = parse_income_spec( + age_min=initial_age, + age_max=final_age, + **income_spec, + SabelhausSong=ss_variances, +) + +# Age groups for the estimation: calculate average wealth-to-permanent income ratio +# for consumers within each of these age groups, compare actual to simulated data + +age_groups = [ + list(range(start, start + age_interval)) + for start in range(initial_age + 1, final_age_data + 1, age_interval) +] + +# generate labels as (25,30], (30,35], ... +age_labels = [f"({group[0]-1},{group[-1]}]" for group in age_groups] + +# Generate mappings between the real ages in the groups and the indices of simulated data +age_mapping = dict(zip(age_labels, map(np.array, age_groups))) +sim_mapping = { + label: np.array(group) - initial_age for label, group in zip(age_labels, age_groups) +} + +remove_ages_from_scf = np.arange( + retirement_age - age_interval + 1, + retirement_age + age_interval + 1, +) # remove retirement ages 61-70 +remove_ages_from_snp = np.arange( + retirement_age + age_interval + 1, +) # only match ages 71 and older + + +init_params_options = { + "init_guess": { + "CRRA": init_CRRA, + "DiscFac": init_DiscFac, + "WealthShare": init_WealthShare, + }, + "upper_bounds": { + "CRRA": bounds_CRRA[1], + "DiscFac": bounds_DiscFac[1], + "WealthShare": bounds_WealthShare[1], + }, + "lower_bounds": { + "CRRA": bounds_CRRA[0], + "DiscFac": bounds_DiscFac[0], + "WealthShare": bounds_WealthShare[0], + }, +} + + +# Survival probabilities over the lifecycle +liv_prb = parse_ssa_life_table( + female=False, + min_age=initial_age, + max_age=final_age - 1, + cohort=1960, +) + +aNrmInit = DiscreteDistribution( + prob_w_to_y, + init_w_to_y, + seed=seed, +).draw(N=num_agents) + +bootstrap_options = { + "bootstrap_size": bootstrap_size, + "seed": seed, +} + +minimize_options = { + "algorithm": "scipy_neldermead", + "multistart": True, + "error_handling": "continue", + "algo_options": { + "convergence.absolute_params_tolerance": 1e-3, + "convergence.absolute_criterion_tolerance": 1e-3, + "stopping.max_iterations": 50, + "stopping.max_criterion_evaluations": 100, + # "n_cores": 12, + }, + # "numdiff_options": {"n_cores": 12}, +} + +# ----------------------------------------------------------------------------- +# -- Set up the dictionary "container" for making a basic lifecycle type ------ +# ----------------------------------------------------------------------------- + +# Dictionary that can be passed to ConsumerType to instantiate +init_calibration = { + **init_lifecycle, + "CRRA": init_CRRA, + "DiscFac": init_DiscFac, + "Rfree": Rfree, + "PermGroFac": inc_calib["PermGroFac"], + "PermGroFacAgg": 1.0, + "BoroCnstArt": BoroCnstArt, + "PermShkStd": inc_calib["PermShkStd"], + "PermShkCount": PermShkCount, + "TranShkStd": inc_calib["TranShkStd"], + "TranShkCount": TranShkCount, + "T_cycle": terminal_t, + "UnempPrb": UnempPrb, + "UnempPrbRet": UnempPrbRet, + "T_retire": retirement_t, + "T_age": terminal_t, + "IncUnemp": IncUnemp, + "IncUnempRet": IncUnempRet, + "aXtraMin": aXtraMin, + "aXtraMax": aXtraMax, + "aXtraCount": aXtraCount, + "aXtraNestFac": exp_nest, + "LivPrb": liv_prb, + "AgentCount": num_agents, + "seed": seed, + "tax_rate": 0.0, + "vFuncBool": vFuncBool, + "CubicBool": CubicBool, + "aNrmInit": aNrmInit, +} + +# from Mateo's JMP for College Educated +ElnR = 0.020 +VlnR = 0.424**2 + +TrueElnR = 0.085 +TrueVlnR = 0.170**2 + +init_subjective_stock = { + "Rfree": 1.019, # from Mateo's JMP + "RiskyAvg": np.exp(ElnR + 0.5 * VlnR), + "RiskyStd": np.sqrt(np.exp(2 * ElnR + VlnR) * (np.exp(VlnR) - 1)), + "RiskyAvgTrue": np.exp(TrueElnR + 0.5 * TrueVlnR), + "RiskyStdTrue": np.sqrt(np.exp(2 * TrueElnR + TrueVlnR) * (np.exp(TrueVlnR) - 1)), +} + +# from Tao's JMP +init_subjective_labor = { + "TranShkStd": [0.03] * len(inc_calib["TranShkStd"]), + "PermShkStd": [0.03] * len(inc_calib["PermShkStd"]), +} diff --git a/code/estimark/msm.py b/code/estimark/msm.py index 3134352..0e1cbe3 100644 --- a/code/estimark/msm.py +++ b/code/estimark/msm.py @@ -34,6 +34,8 @@ def estimate_msm( params_to_estimate=None, subjective_stock=False, subjective_labor=False, + emp_moments=None, + moments_cov=None, ): ############################################################ # Make agent @@ -48,26 +50,32 @@ def estimate_msm( print("Agent created: ", agent.name) ############################################################ - # Get empirical moments + # Get initial guess ############################################################ - emp_moments = get_empirical_moments(agent.name) + initial_guess = get_initial_guess(agent.name, params_to_estimate, tables_dir) - print("Calculated empirical moments.") + print(initial_guess) ############################################################ - # Get moments covariance matrix + # Get empirical moments ############################################################ - moments_cov = get_moments_cov(agent.name, emp_moments) + if emp_moments is None: + + emp_moments = get_empirical_moments(agent.name) - print("Calculated moments covariance matrix.") + print("Calculated empirical moments.") ############################################################ - # Get initial guess + # Get moments covariance matrix ############################################################ - initial_guess = get_initial_guess(agent.name, params_to_estimate, tables_dir) + if moments_cov is None: + + moments_cov = get_moments_cov(agent.name, emp_moments) + + print("Calculated moments covariance matrix.") print("Estimating MSM...") @@ -106,13 +114,13 @@ def estimate_msm( if __name__ == "__main__": # Set booleans to determine which tasks should be done # Which agent type to estimate ("IndShock" or "Portfolio") - local_agent_name = "Portfolio" + local_agent_name = "WealthPortfolio" # Whether to use subjective beliefs local_subjective_stock = False local_subjective_labor = False - local_params_to_estimate = ["CRRA", "DiscFac"] + local_params_to_estimate = ["CRRA", "DiscFac", "WealthShare"] estimate_msm( init_agent_name=local_agent_name, diff --git a/code/estimark/parameters.py b/code/estimark/parameters.py index 689c420..de290fc 100644 --- a/code/estimark/parameters.py +++ b/code/estimark/parameters.py @@ -62,10 +62,13 @@ # Initial guess of the adjustment to the discount factor during estimation (beth) init_DiscFac = 0.95 # Bounds for beth; if violated, objective function returns "penalty value" -bounds_DiscFac = [0.5, 1.0] +bounds_DiscFac = [0.5, 1.1] # Bounds for rho; if violated, objective function returns "penalty value" bounds_CRRA = [1.1, 20.0] +init_WealthShare = 0.5 # Initial guess of the wealth share parameter +bounds_WealthShare = [0.0, 1.0] # Bounds for the wealth share parameter + ###################################################################### # Constructed parameters @@ -113,9 +116,21 @@ init_params_options = { - "init_guess": {"CRRA": init_CRRA, "DiscFac": init_DiscFac}, - "upper_bounds": {"CRRA": bounds_CRRA[1], "DiscFac": bounds_DiscFac[1]}, - "lower_bounds": {"CRRA": bounds_CRRA[0], "DiscFac": bounds_DiscFac[0]}, + "init_guess": { + "CRRA": init_CRRA, + "DiscFac": init_DiscFac, + "WealthShare": init_WealthShare, + }, + "upper_bounds": { + "CRRA": bounds_CRRA[1], + "DiscFac": bounds_DiscFac[1], + "WealthShare": bounds_WealthShare[1], + }, + "lower_bounds": { + "CRRA": bounds_CRRA[0], + "DiscFac": bounds_DiscFac[0], + "WealthShare": bounds_WealthShare[0], + }, } diff --git a/code/run_all_msm.py b/code/run_all_msm.py new file mode 100644 index 0000000..8f42563 --- /dev/null +++ b/code/run_all_msm.py @@ -0,0 +1,52 @@ +from estimark.msm import estimate_msm +from estimark.estimation import get_empirical_moments, get_moments_cov + + +if __name__ == "__main__": + + emp_moments = get_empirical_moments("") + port_emp_moments = get_empirical_moments("Porfolio") + + moments_cov = get_moments_cov("", emp_moments) + port_moments_cov = get_moments_cov("Porfolio", port_emp_moments) + + for i in range(1, 6): + which_model = str(i) + for k in range(1, 5): + subjective_markets = str(k) + + replication_specs = {} + + if which_model == "1" or which_model == "": + replication_specs["init_agent_name"] = "IndShock" + elif which_model == "2": + replication_specs["init_agent_name"] = "Portfolio" + elif which_model == "3": + replication_specs["init_agent_name"] = "WarmGlow" + elif which_model == "4": + replication_specs["init_agent_name"] = "WarmGlowPortfolio" + elif which_model == "5": + replication_specs["init_agent_name"] = "WealthPortfolio" + + print("Model: ", replication_specs["init_agent_name"]) + + replication_specs["params_to_estimate"] = ["CRRA", "DiscFac"] + + if subjective_markets == "2" or subjective_markets == "4": + replication_specs["subjective_stock"] = True + print("Adding subjective stock market beliefs...") + + if subjective_markets == "3" or subjective_markets == "4": + replication_specs["subjective_labor"] = True + print("Adding subjective labor market beliefs...") + + if "Portfolio" in replication_specs["init_agent_name"]: + replication_specs["emp_moments"] = port_emp_moments + replication_specs["moments_cov"] = port_moments_cov + else: + replication_specs["emp_moments"] = emp_moments + replication_specs["moments_cov"] = moments_cov + + estimate_msm(**replication_specs) + + print("All replications complete.") diff --git a/content/tables/msm/IndShockSub(Labor)Market_estimate_results.csv b/content/tables/msm/IndShockSub(Labor)Market_estimate_results.csv new file mode 100644 index 0000000..4758098 --- /dev/null +++ b/content/tables/msm/IndShockSub(Labor)Market_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,10.120079274781993 +DiscFac,1.072981483284301 +time_to_estimate,443.3000178337097 diff --git a/content/tables/msm/IndShockSub(Stock)(Labor)Market_estimate_results.csv b/content/tables/msm/IndShockSub(Stock)(Labor)Market_estimate_results.csv new file mode 100644 index 0000000..1614f36 --- /dev/null +++ b/content/tables/msm/IndShockSub(Stock)(Labor)Market_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,10.585905966806981 +DiscFac,1.075380207950192 +time_to_estimate,382.08420753479004 diff --git a/content/tables/msm/IndShockSub(Stock)Market_estimate_results.csv b/content/tables/msm/IndShockSub(Stock)Market_estimate_results.csv new file mode 100644 index 0000000..b091278 --- /dev/null +++ b/content/tables/msm/IndShockSub(Stock)Market_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,1.6900656867736041 +DiscFac,0.9795057942445009 +time_to_estimate,385.40881276130676 diff --git a/content/tables/msm/IndShock_estimate_results.csv b/content/tables/msm/IndShock_estimate_results.csv index 2792138..d28f3db 100644 --- a/content/tables/msm/IndShock_estimate_results.csv +++ b/content/tables/msm/IndShock_estimate_results.csv @@ -1,3 +1,3 @@ -CRRA,1.7137312834076943 -DiscFac,0.9711633132004429 -time_to_estimate,310.4867904186249 +CRRA,1.7355613236768712 +DiscFac,0.9705704731287197 +time_to_estimate,349.9392056465149 diff --git a/content/tables/msm/PortfolioSub(Labor)Market_estimate_results.csv b/content/tables/msm/PortfolioSub(Labor)Market_estimate_results.csv new file mode 100644 index 0000000..e8e9ff4 --- /dev/null +++ b/content/tables/msm/PortfolioSub(Labor)Market_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,15.796329213228365 +DiscFac,1.01881362645317 +time_to_estimate,530.7010610103607 diff --git a/content/tables/msm/PortfolioSub(Stock)(Labor)Market_estimate_results.csv b/content/tables/msm/PortfolioSub(Stock)(Labor)Market_estimate_results.csv new file mode 100644 index 0000000..ed6488c --- /dev/null +++ b/content/tables/msm/PortfolioSub(Stock)(Labor)Market_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,10.443858291162275 +DiscFac,1.0472068922536986 +time_to_estimate,432.9137485027313 diff --git a/content/tables/msm/PortfolioSub(Stock)Market_estimate_results.csv b/content/tables/msm/PortfolioSub(Stock)Market_estimate_results.csv new file mode 100644 index 0000000..a39e3a1 --- /dev/null +++ b/content/tables/msm/PortfolioSub(Stock)Market_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,2.028141210825867 +DiscFac,0.9475017000972952 +time_to_estimate,366.9574282169342 diff --git a/content/tables/msm/Portfolio_estimate_results.csv b/content/tables/msm/Portfolio_estimate_results.csv index 10bd388..3456bc9 100644 --- a/content/tables/msm/Portfolio_estimate_results.csv +++ b/content/tables/msm/Portfolio_estimate_results.csv @@ -1,3 +1,3 @@ -CRRA,7.988151380338656 -DiscFac,0.7012415354852454 -time_to_estimate,247.5582902431488 +CRRA,7.937396585496538 +DiscFac,0.7340929722871067 +time_to_estimate,387.3447730541229 diff --git a/content/tables/msm/WarmGlowPortfolioSub(Labor)Market_estimate_results.csv b/content/tables/msm/WarmGlowPortfolioSub(Labor)Market_estimate_results.csv new file mode 100644 index 0000000..a4db41e --- /dev/null +++ b/content/tables/msm/WarmGlowPortfolioSub(Labor)Market_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,15.801949476432116 +DiscFac,1.0088188084108516 +time_to_estimate,513.6281001567841 diff --git a/content/tables/msm/WarmGlowPortfolioSub(Stock)(Labor)Market_estimate_results.csv b/content/tables/msm/WarmGlowPortfolioSub(Stock)(Labor)Market_estimate_results.csv new file mode 100644 index 0000000..d0662c3 --- /dev/null +++ b/content/tables/msm/WarmGlowPortfolioSub(Stock)(Labor)Market_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,10.482525425704559 +DiscFac,1.046890433862215 +time_to_estimate,398.2840714454651 diff --git a/content/tables/msm/WarmGlowPortfolioSub(Stock)Market_estimate_results.csv b/content/tables/msm/WarmGlowPortfolioSub(Stock)Market_estimate_results.csv new file mode 100644 index 0000000..6e3332a --- /dev/null +++ b/content/tables/msm/WarmGlowPortfolioSub(Stock)Market_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,2.028141210825867 +DiscFac,0.9475017000972952 +time_to_estimate,353.6740868091583 diff --git a/content/tables/msm/WarmGlowPortfolio_estimate_results.csv b/content/tables/msm/WarmGlowPortfolio_estimate_results.csv new file mode 100644 index 0000000..3973ec7 --- /dev/null +++ b/content/tables/msm/WarmGlowPortfolio_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,8.136028289794922 +DiscFac,0.7272434997558592 +time_to_estimate,358.84484124183655 diff --git a/content/tables/msm/WarmGlowSub(Labor)Market_estimate_results.csv b/content/tables/msm/WarmGlowSub(Labor)Market_estimate_results.csv new file mode 100644 index 0000000..e825a79 --- /dev/null +++ b/content/tables/msm/WarmGlowSub(Labor)Market_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,4.351901149954774 +DiscFac,1.030069934355222 +time_to_estimate,522.19313788414 diff --git a/content/tables/msm/WarmGlowSub(Stock)(Labor)Market_estimate_results.csv b/content/tables/msm/WarmGlowSub(Stock)(Labor)Market_estimate_results.csv new file mode 100644 index 0000000..b1166af --- /dev/null +++ b/content/tables/msm/WarmGlowSub(Stock)(Labor)Market_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,5.377015394317432 +DiscFac,1.0436200618163325 +time_to_estimate,519.4643342494965 diff --git a/content/tables/msm/WarmGlowSub(Stock)Market_estimate_results.csv b/content/tables/msm/WarmGlowSub(Stock)Market_estimate_results.csv new file mode 100644 index 0000000..494ffef --- /dev/null +++ b/content/tables/msm/WarmGlowSub(Stock)Market_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,1.606618169262997 +DiscFac,0.9790262793856838 +time_to_estimate,366.2848656177521 diff --git a/content/tables/msm/WarmGlow_estimate_results.csv b/content/tables/msm/WarmGlow_estimate_results.csv new file mode 100644 index 0000000..809f82c --- /dev/null +++ b/content/tables/msm/WarmGlow_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,1.6965301440096874 +DiscFac,0.9694670056908944 +time_to_estimate,381.16720819473267 diff --git a/content/tables/msm/WealthPortfolioSub(Labor)Market_estimate_results.csv b/content/tables/msm/WealthPortfolioSub(Labor)Market_estimate_results.csv new file mode 100644 index 0000000..0d97511 --- /dev/null +++ b/content/tables/msm/WealthPortfolioSub(Labor)Market_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,7.903817069560471 +DiscFac,0.9925572008651333 +time_to_estimate,523.6651594638824 diff --git a/content/tables/msm/WealthPortfolioSub(Stock)(Labor)Market_estimate_results.csv b/content/tables/msm/WealthPortfolioSub(Stock)(Labor)Market_estimate_results.csv new file mode 100644 index 0000000..2d921a4 --- /dev/null +++ b/content/tables/msm/WealthPortfolioSub(Stock)(Labor)Market_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,8.150800171126672 +DiscFac,1.0042295369139171 +time_to_estimate,500.8115863800049 diff --git a/content/tables/msm/WealthPortfolioSub(Stock)Market_estimate_results.csv b/content/tables/msm/WealthPortfolioSub(Stock)Market_estimate_results.csv new file mode 100644 index 0000000..7f6cbce --- /dev/null +++ b/content/tables/msm/WealthPortfolioSub(Stock)Market_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,4.5045093358857615 +DiscFac,0.8386902862207641 +time_to_estimate,426.64623165130615 diff --git a/content/tables/msm/WealthPortfolio_estimate_results.csv b/content/tables/msm/WealthPortfolio_estimate_results.csv new file mode 100644 index 0000000..c14740d --- /dev/null +++ b/content/tables/msm/WealthPortfolio_estimate_results.csv @@ -0,0 +1,3 @@ +CRRA,4.566021017209886 +DiscFac,0.8273955783588784 +time_to_estimate,350.6654374599457