policyengine-us 1.425.7__py3-none-any.whl → 1.426.0__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (21) hide show
  1. policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/README.md +206 -1
  2. policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/income/age_multiplier_over_threshold.yaml +11 -0
  3. policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/income/age_threshold.yaml +9 -0
  4. policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/{income.yaml → income/all.yaml} +1 -1
  5. policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/income/base.yaml +11 -0
  6. policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/substitution/age_multiplier_over_threshold.yaml +11 -0
  7. policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/substitution/age_threshold.yaml +9 -0
  8. policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/substitution/all.yaml +6 -0
  9. policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/substitution/by_position_and_decile.yaml +71 -0
  10. policyengine_us/tests/policy/baseline/gov/simulation/labor_supply_response/income_elasticity.yaml +12 -6
  11. policyengine_us/tests/policy/baseline/gov/simulation/labor_supply_response/integration.yaml +45 -0
  12. policyengine_us/tests/policy/baseline/gov/simulation/labor_supply_response/labor_supply_behavioral_response.yaml +2 -24
  13. policyengine_us/tests/policy/baseline/gov/simulation/labor_supply_response/substitution_elasticity.yaml +10 -35
  14. policyengine_us/variables/gov/simulation/labor_supply_response/income_elasticity.py +27 -1
  15. policyengine_us/variables/gov/simulation/labor_supply_response/substitution_elasticity.py +43 -18
  16. {policyengine_us-1.425.7.dist-info → policyengine_us-1.426.0.dist-info}/METADATA +1 -1
  17. {policyengine_us-1.425.7.dist-info → policyengine_us-1.426.0.dist-info}/RECORD +20 -13
  18. policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/substitution.yaml +0 -70
  19. {policyengine_us-1.425.7.dist-info → policyengine_us-1.426.0.dist-info}/WHEEL +0 -0
  20. {policyengine_us-1.425.7.dist-info → policyengine_us-1.426.0.dist-info}/entry_points.txt +0 -0
  21. {policyengine_us-1.425.7.dist-info → policyengine_us-1.426.0.dist-info}/licenses/LICENSE +0 -0
@@ -1 +1,206 @@
1
- # Elasticities
1
+ # Labor Supply Response Elasticities
2
+
3
+ This directory contains parameters for labor supply response elasticities used in behavioral microsimulation analysis.
4
+
5
+ ## Overview
6
+
7
+ Labor supply elasticities measure how individuals adjust their work behavior in response to changes in economic incentives:
8
+
9
+ - **Substitution elasticity**: How labor supply responds to changes in the effective marginal wage rate (after-tax wage)
10
+ - **Income elasticity**: How labor supply responds to changes in disposable income
11
+
12
+ ## Age Heterogeneity: The Multiplier Approach
13
+
14
+ Both elasticity types support age-based heterogeneity through an **age multiplier** that scales base elasticities for individuals aged 65 and over. This approach is based on empirical research showing that older workers have higher labor supply elasticities than working-age adults.
15
+
16
+ ### Research Findings
17
+
18
+ - **French (2005)**: Elasticities 3.0x-3.25x higher for age 60 vs age 40 workers
19
+ - **CBO Working Papers (2012-12, 2012-13)**: Frisch elasticity ranges from 0.27 to 0.53 (central: 0.40) for working-age adults
20
+ - **General pattern**: Retirement-age individuals (62-70) have particularly high elasticities, especially on the extensive margin (whether to work at all)
21
+
22
+ ### Why a Multiplier Approach?
23
+
24
+ The multiplier approach was chosen over separate age-specific parameters because:
25
+ 1. **Evidence-based**: Literature consistently shows older workers are more elastic, but doesn't provide income-decile-specific multipliers
26
+ 2. **Parsimony**: 13 total parameters instead of 44 (11 base elasticities + 2 age multipliers)
27
+ 3. **Transparency**: Users can easily understand and adjust one multiplier value
28
+ 4. **Flexibility**: Multiplier can range from 1.0 (no age effect) to 3.0+ (French 2005 finding)
29
+
30
+ ## Substitution Elasticity Structure
31
+
32
+ The substitution elasticity uses a two-step calculation:
33
+
34
+ ### Step 1: Base Elasticity (by position and decile)
35
+
36
+ Base elasticities vary by:
37
+ 1. **Position**: primary earner vs secondary earner within tax unit
38
+ 2. **Decile**: 10 income deciles (for primary earners only)
39
+
40
+ This creates 11 base elasticity parameters:
41
+ - 10 for primary earners by decile
42
+ - 1 for secondary earners (all deciles)
43
+
44
+ ### Step 2: Age Multiplier
45
+
46
+ For individuals aged 65 and over, the base elasticity is multiplied by `age_multiplier_65_and_over`.
47
+
48
+ **Formula**:
49
+ - If age < 65: `elasticity = base_elasticity`
50
+ - If age >= 65: `elasticity = base_elasticity × age_multiplier_65_and_over`
51
+
52
+ ### Global Override
53
+
54
+ The `all` parameter overrides all base elasticities and age multipliers if set to non-zero.
55
+
56
+ ## Income Elasticity Structure
57
+
58
+ The income elasticity uses a simpler two-parameter structure:
59
+
60
+ 1. **Base elasticity**: Applied to all working-age individuals (under 65)
61
+ 2. **Age multiplier**: Applied to individuals 65 and over
62
+
63
+ **Formula**:
64
+ - If age < 65: `elasticity = base`
65
+ - If age >= 65: `elasticity = base × age_multiplier_65_and_over`
66
+
67
+ The `all` parameter overrides base and multiplier if set to non-zero.
68
+
69
+ ## Default Values
70
+
71
+ - All base elasticities default to 0 (no behavioral response)
72
+ - Age multipliers default to 2.0 (conservative estimate based on literature)
73
+ - Users must explicitly set base elasticity values to enable behavioral responses
74
+
75
+ ## Usage Examples
76
+
77
+ ### Example 1: No behavioral response (default)
78
+ ```yaml
79
+ substitution:
80
+ by_position_and_decile:
81
+ primary:
82
+ 1: 0
83
+ 2: 0
84
+ # ... all zeros
85
+ secondary: 0
86
+ age_multiplier_65_and_over: 2.0 # Doesn't matter since base is 0
87
+
88
+ income:
89
+ base: 0
90
+ age_multiplier_65_and_over: 2.0 # Doesn't matter since base is 0
91
+ ```
92
+
93
+ Result: Everyone has zero elasticity regardless of age.
94
+
95
+ ### Example 2: Same elasticity for all ages
96
+ ```yaml
97
+ substitution:
98
+ by_position_and_decile:
99
+ primary:
100
+ 1: 0.31 # CBO-style values
101
+ 2: 0.27
102
+ # ... other deciles
103
+ secondary: 0.40
104
+ age_multiplier_65_and_over: 1.0 # No age effect
105
+
106
+ income:
107
+ base: -0.04
108
+ age_multiplier_65_and_over: 1.0 # No age effect
109
+ ```
110
+
111
+ Result: Same elasticity for everyone regardless of age.
112
+
113
+ ### Example 3: Higher elasticity for elderly (RECOMMENDED)
114
+ ```yaml
115
+ substitution:
116
+ by_position_and_decile:
117
+ primary:
118
+ 1: 0.31
119
+ 2: 0.27
120
+ # ... other deciles
121
+ secondary: 0.40
122
+ age_multiplier_65_and_over: 2.0 # Conservative estimate
123
+
124
+ income:
125
+ base: -0.04
126
+ age_multiplier_65_and_over: 2.0
127
+ ```
128
+
129
+ Result:
130
+ - Age 40, primary earner, decile 1: elasticity = 0.31
131
+ - Age 70, primary earner, decile 1: elasticity = 0.31 × 2.0 = 0.62
132
+ - Age 40: income elasticity = -0.04
133
+ - Age 70: income elasticity = -0.04 × 2.0 = -0.08
134
+
135
+ ### Example 4: Aggressive multiplier based on French (2005)
136
+ ```yaml
137
+ substitution:
138
+ age_multiplier_65_and_over: 3.0 # Upper end of literature
139
+
140
+ income:
141
+ base: -0.04
142
+ age_multiplier_65_and_over: 3.0
143
+ ```
144
+
145
+ Result:
146
+ - Age 70, primary earner, decile 1 (base 0.31): elasticity = 0.31 × 3.0 = 0.93
147
+ - Age 70: income elasticity = -0.04 × 3.0 = -0.12
148
+
149
+ ## Multiplier Value Guidance
150
+
151
+ Based on literature review (see `age_heterogeneity_analysis.md`):
152
+
153
+ | Multiplier | Interpretation | Source |
154
+ |------------|----------------|--------|
155
+ | 1.0 | No age difference | Ignores evidence |
156
+ | 1.5 | Conservative | Lower end of estimates |
157
+ | 2.0 | **Recommended default** | Conservative but evidence-based |
158
+ | 2.5 | Moderate | Midpoint of French (2005) range |
159
+ | 3.0 | Aggressive | French (2005) finding for age 60 vs 40 |
160
+ | 3.25 | Upper bound | Upper end of French (2005) range |
161
+
162
+ **Note**: The exact multiplier value has uncertainty. We recommend starting with 2.0 and conducting sensitivity analysis with values in the 1.5-3.0 range.
163
+
164
+ ## References
165
+
166
+ - [French (2005): "The Effects of Health, Wealth, and Wages on Labour Supply and Retirement Behaviour"](https://academic.oup.com/restud/article-abstract/72/2/395/1558553) - Review of Economic Studies 72(2): 395-427
167
+ - [CBO Working Paper 2012-12: "A Review of Recent Research on Labor Supply Elasticities"](https://www.cbo.gov/publication/43675) - McClelland & Mok
168
+ - [CBO Working Paper 2012-13: "Review of Estimates of the Frisch Elasticity of Labor Supply"](https://www.cbo.gov/publication/43676) - Reichling & Whalen
169
+
170
+ ## Implementation Notes
171
+
172
+ - Age is determined from the `age` variable for the year (`period.this_year`)
173
+ - Age 65 is the cutoff: age < 65 uses base elasticity, age >= 65 applies multiplier
174
+ - Primary earner is defined as the highest earner within the tax unit
175
+ - Earnings deciles are determined using hardcoded markers (TODO: parametrize)
176
+ - Negative total earnings result in zero elasticity (using `max_(earnings, 0)`)
177
+ - Zero base elasticity remains zero even with multiplier (0 × multiplier = 0)
178
+
179
+ ## Technical Details
180
+
181
+ ### How Primary/Secondary Earner is Determined
182
+
183
+ Within each tax unit, the person with the highest total earnings (employment + self-employment) is designated as the primary earner. All other earners in the unit are secondary earners. This means:
184
+ - Single-person tax units: That person is always primary
185
+ - Multi-person tax units: Only the highest earner gets primary elasticity; others get zero (since secondary earner base elasticities default to 0)
186
+
187
+ ### Earnings Decile Markers
188
+
189
+ Current hardcoded decile boundaries (TODO: parametrize):
190
+ - Decile 1: $0 - $14,000
191
+ - Decile 2: $14,000 - $28,000
192
+ - Decile 3: $28,000 - $39,000
193
+ - Decile 4: $39,000 - $50,000
194
+ - Decile 5: $50,000 - $61,000
195
+ - Decile 6: $61,000 - $76,000
196
+ - Decile 7: $76,000 - $97,000
197
+ - Decile 8: $97,000 - $138,000
198
+ - Decile 9: $138,000 - $1,726,000
199
+ - Decile 10: $1,726,000+
200
+
201
+ ### Negative Earnings Handling
202
+
203
+ To prevent negative earnings from causing sign flips in economic responses, the code applies `max_(earnings, 0)` before calculating elasticities. This means:
204
+ - Positive net earnings: Normal elasticity calculation
205
+ - Negative net earnings: Treated as zero earnings, resulting in zero substitution elasticity
206
+ - Income elasticity still applies to individuals with negative earnings
@@ -0,0 +1,11 @@
1
+ description: Multiplier applied to base income elasticity for individuals at or above the age threshold. Research suggests income effects are stronger for older workers. French (2005) shows much higher elasticities near retirement age. A multiplier of 2.0 is conservative. Set to 1.0 for no age difference.
2
+ values:
3
+ 2020-01-01: 2.0
4
+ metadata:
5
+ unit: /1
6
+ label: elasticity multiplier for ages at or above threshold
7
+ reference:
8
+ - title: "French (2005): The Effects of Health, Wealth, and Wages on Labour Supply and Retirement Behaviour"
9
+ href: https://academic.oup.com/restud/article-abstract/72/2/395/1558553
10
+ - title: "CBO Working Paper 2012-12: A Review of Recent Research on Labor Supply Elasticities"
11
+ href: https://www.cbo.gov/publication/43675
@@ -0,0 +1,9 @@
1
+ description: Age at which the age multiplier begins to apply. Individuals at this age or older will have their base elasticity multiplied by the age multiplier. Default is 65 (typical retirement age), but can be adjusted to reflect different retirement patterns or policy scenarios (e.g., 62 for early retirement, 67 for full retirement age).
2
+ values:
3
+ 2020-01-01: 65
4
+ metadata:
5
+ unit: year
6
+ label: age threshold for elasticity multiplier
7
+ reference:
8
+ - title: "French (2005): The Effects of Health, Wealth, and Wages on Labour Supply and Retirement Behaviour"
9
+ href: https://academic.oup.com/restud/article-abstract/72/2/395/1558553
@@ -1,4 +1,4 @@
1
- description: Percent change (of the change in disposable income) in labor supply given a 1% change in disposable income.
1
+ description: Percent change (of the change in disposable income) in labor supply given a 1% change in disposable income. This parameter overrides the base income elasticity and age multiplier if provided.
2
2
  values:
3
3
  2020-01-01: 0
4
4
  metadata:
@@ -0,0 +1,11 @@
1
+ description: Base income elasticity of labor supply for working-age individuals (under 65). This value is multiplied by the age multiplier for individuals 65 and over. Typically negative, indicating that higher income reduces labor supply.
2
+ values:
3
+ 2020-01-01: 0
4
+ metadata:
5
+ unit: /1
6
+ label: base income elasticity below threshold
7
+ reference:
8
+ - title: "CBO Working Paper 2012-12: A Review of Recent Research on Labor Supply Elasticities"
9
+ href: https://www.cbo.gov/publication/43675
10
+ - title: "CBO Working Paper 2012-13: Review of Estimates of the Frisch Elasticity of Labor Supply"
11
+ href: https://www.cbo.gov/publication/43676
@@ -0,0 +1,11 @@
1
+ description: Multiplier applied to all substitution elasticities for individuals at or above the age threshold. Research shows older workers have higher labor supply elasticities than working-age adults. French (2005) finds elasticities 3x higher for age 60 vs age 40. A multiplier of 2.0 is conservative. Set to 1.0 for no age difference.
2
+ values:
3
+ 2020-01-01: 2.0
4
+ metadata:
5
+ unit: /1
6
+ label: elasticity multiplier for ages at or above threshold
7
+ reference:
8
+ - title: "French (2005): The Effects of Health, Wealth, and Wages on Labour Supply and Retirement Behaviour"
9
+ href: https://academic.oup.com/restud/article-abstract/72/2/395/1558553
10
+ - title: "CBO Working Paper 2012-12: A Review of Recent Research on Labor Supply Elasticities"
11
+ href: https://www.cbo.gov/publication/43675
@@ -0,0 +1,9 @@
1
+ description: Age at which the age multiplier begins to apply. Individuals at this age or older will have their base elasticity multiplied by the age multiplier. Default is 65 (typical retirement age), but can be adjusted to reflect different retirement patterns or policy scenarios (e.g., 62 for early retirement, 67 for full retirement age).
2
+ values:
3
+ 2020-01-01: 65
4
+ metadata:
5
+ unit: year
6
+ label: age threshold for elasticity multiplier
7
+ reference:
8
+ - title: "French (2005): The Effects of Health, Wealth, and Wages on Labour Supply and Retirement Behaviour"
9
+ href: https://academic.oup.com/restud/article-abstract/72/2/395/1558553
@@ -0,0 +1,6 @@
1
+ description: Percent change (of the change in the effective marginal wage) in labor supply given a 1% change in the effective marginal wage. This parameter overrides all other substitution elasticities if provided.
2
+ values:
3
+ 2020-01-01: 0
4
+ metadata:
5
+ unit: /1
6
+ label: substitution elasticity of labor supply
@@ -0,0 +1,71 @@
1
+ metadata:
2
+ label: by position and decile
3
+ unit: /1
4
+ propagate_metadata_to_children: true
5
+ reference:
6
+ - title: "CBO Working Paper 2012-12: A Review of Recent Research on Labor Supply Elasticities"
7
+ href: https://www.cbo.gov/publication/43675
8
+ - title: "CBO Working Paper 2012-13: Review of Estimates of the Frisch Elasticity of Labor Supply"
9
+ href: https://www.cbo.gov/publication/43676
10
+ - title: "French (2005): The Effects of Health, Wealth, and Wages on Labour Supply and Retirement Behaviour"
11
+ href: https://academic.oup.com/restud/article-abstract/72/2/395/1558553
12
+ primary:
13
+ metadata:
14
+ label: primary adult
15
+ description: The highest earner within the tax unit. Elasticity varies by earnings decile.
16
+ 1:
17
+ metadata:
18
+ label: primary adult, 1st decile substitution elasticity
19
+ values:
20
+ 2020-01-01: 0
21
+ 2:
22
+ metadata:
23
+ label: primary adult, 2nd decile substitution elasticity
24
+ values:
25
+ 2020-01-01: 0
26
+ 3:
27
+ metadata:
28
+ label: primary adult, 3rd decile substitution elasticity
29
+ values:
30
+ 2020-01-01: 0
31
+ 4:
32
+ metadata:
33
+ label: primary adult, 4th decile substitution elasticity
34
+ values:
35
+ 2020-01-01: 0
36
+ 5:
37
+ metadata:
38
+ label: primary adult, 5th decile substiticity
39
+ values:
40
+ 2020-01-01: 0
41
+ 6:
42
+ metadata:
43
+ label: primary adult, 6th decile substitution elasticity
44
+ values:
45
+ 2020-01-01: 0
46
+ 7:
47
+ metadata:
48
+ label: primary adult, 7th decile substitution elasticity
49
+ values:
50
+ 2020-01-01: 0
51
+ 8:
52
+ metadata:
53
+ label: primary adult, 8th decile substitution elasticity
54
+ values:
55
+ 2020-01-01: 0
56
+ 9:
57
+ metadata:
58
+ label: primary adult, 9th decile substitution elasticity
59
+ values:
60
+ 2020-01-01: 0
61
+ 10:
62
+ metadata:
63
+ label: primary adult, 10th decile substitution elasticity
64
+ values:
65
+ 2020-01-01: 0
66
+ secondary:
67
+ metadata:
68
+ label: secondary adult, all deciles substitution elasticity
69
+ description: Any earner in the tax unit who is not the primary (highest) earner.
70
+ values:
71
+ 2020-01-01: 0
@@ -1,13 +1,19 @@
1
- - name: Default income elasticity is zero
1
+ - name: Income elasticity for person under 65 uses base
2
2
  period: 2023
3
3
  input:
4
- gov.simulation.labor_supply_responses.elasticities.income: 0
4
+ age: 50
5
+ gov.simulation.labor_supply_responses.elasticities.income.all: 0
6
+ gov.simulation.labor_supply_responses.elasticities.income.base: -0.04
7
+ gov.simulation.labor_supply_responses.elasticities.income.age_multiplier_over_threshold: 2.0
5
8
  output:
6
- income_elasticity: 0
9
+ income_elasticity: -0.04
7
10
 
8
- - name: Custom income elasticity value
11
+ - name: Income elasticity for person 65+ applies multiplier
9
12
  period: 2023
10
13
  input:
11
- gov.simulation.labor_supply_responses.elasticities.income: -0.2
14
+ age: 70
15
+ gov.simulation.labor_supply_responses.elasticities.income.all: 0
16
+ gov.simulation.labor_supply_responses.elasticities.income.base: -0.04
17
+ gov.simulation.labor_supply_responses.elasticities.income.age_multiplier_over_threshold: 2.0
12
18
  output:
13
- income_elasticity: -0.2
19
+ income_elasticity: -0.08
@@ -0,0 +1,45 @@
1
+ - name: Couple with different ages (under 65 and 65+) with multiplier approach
2
+ period: 2023
3
+ input:
4
+ people:
5
+ person1:
6
+ age: 67
7
+ employment_income_before_lsr: 60_000
8
+ self_employment_income_before_lsr: 0
9
+ person2:
10
+ age: 55
11
+ employment_income_before_lsr: 40_000
12
+ self_employment_income_before_lsr: 0
13
+ tax_units:
14
+ tax_unit:
15
+ members: [person1, person2]
16
+ gov.simulation.labor_supply_responses.elasticities.substitution.all: 0
17
+ gov.simulation.labor_supply_responses.elasticities.substitution.by_position_and_decile.primary.5: 0.20
18
+ gov.simulation.labor_supply_responses.elasticities.substitution.by_position_and_decile.secondary: 0.30
19
+ gov.simulation.labor_supply_responses.elasticities.substitution.age_multiplier_over_threshold: 2.0
20
+ gov.simulation.labor_supply_responses.elasticities.income.all: 0
21
+ gov.simulation.labor_supply_responses.elasticities.income.base: -0.04
22
+ gov.simulation.labor_supply_responses.elasticities.income.age_multiplier_over_threshold: 2.0
23
+ output:
24
+ substitution_elasticity:
25
+ - 0.40 # person1: 67 years old, primary earner (60k), decile 5 (50k-61k), base 0.20 × multiplier 2.0 = 0.40
26
+ - 0 # person2: 55 years old, secondary earner, gets 0 since not primary
27
+ income_elasticity:
28
+ - -0.08 # person1: 67 years old, base -0.04 × multiplier 2.0 = -0.08
29
+ - -0.04 # person2: 55 years old, base -0.04 (no multiplier)
30
+
31
+ - name: Person with negative net earnings has zero substitution elasticity
32
+ period: 2023
33
+ input:
34
+ age: 66
35
+ employment_income_before_lsr: 25_000
36
+ self_employment_income_before_lsr: -30_000
37
+ gov.simulation.labor_supply_responses.elasticities.substitution.all: 0
38
+ gov.simulation.labor_supply_responses.elasticities.substitution.by_position_and_decile.primary.1: 0.25
39
+ gov.simulation.labor_supply_responses.elasticities.substitution.age_multiplier_over_threshold: 2.0
40
+ gov.simulation.labor_supply_responses.elasticities.income.all: 0
41
+ gov.simulation.labor_supply_responses.elasticities.income.base: -0.04
42
+ gov.simulation.labor_supply_responses.elasticities.income.age_multiplier_over_threshold: 2.25
43
+ output:
44
+ substitution_elasticity: 0 # Negative net earnings (-5k) treated as 0, resulting in zero elasticity
45
+ income_elasticity: -0.09 # Income elasticity still applies: -0.04 × 2.25 = -0.09
@@ -1,31 +1,9 @@
1
- - name: No elasticities means no response
2
- period: 2023
3
- input:
4
- employment_income_before_lsr: 50_000
5
- self_employment_income_before_lsr: 10_000
6
- gov.simulation.labor_supply_responses.elasticities.income: 0
7
- gov.simulation.labor_supply_responses.elasticities.substitution.all: 0
8
- output:
9
- labor_supply_behavioral_response: 0
10
-
11
- - name: Zero income and substitution elasticities
12
- period: 2023
13
- input:
14
- employment_income_before_lsr: 50_000
15
- self_employment_income_before_lsr: 10_000
16
- income_elasticity_lsr: 0
17
- substitution_elasticity_lsr: 0
18
- gov.simulation.labor_supply_responses.elasticities.income: 0
19
- gov.simulation.labor_supply_responses.elasticities.substitution.all: 0
20
- output:
21
- labor_supply_behavioral_response: 0
22
-
23
1
  - name: Negative total earnings with elasticities off
24
2
  period: 2023
25
3
  input:
26
4
  employment_income_before_lsr: 30_000
27
5
  self_employment_income_before_lsr: -40_000 # Net negative earnings
28
- gov.simulation.labor_supply_responses.elasticities.income: 0
6
+ gov.simulation.labor_supply_responses.elasticities.income.all: 0
29
7
  gov.simulation.labor_supply_responses.elasticities.substitution.all: 0
30
8
  output:
31
9
  labor_supply_behavioral_response: 0
@@ -35,7 +13,7 @@
35
13
  input:
36
14
  employment_income_before_lsr: 50_000
37
15
  self_employment_income_before_lsr: -10_000 # Net positive earnings = 40_000
38
- gov.simulation.labor_supply_responses.elasticities.income: 0
16
+ gov.simulation.labor_supply_responses.elasticities.income.all: 0
39
17
  gov.simulation.labor_supply_responses.elasticities.substitution.all: 0
40
18
  output:
41
19
  labor_supply_behavioral_response: 0
@@ -1,48 +1,23 @@
1
- - name: Default substitution elasticity is zero
1
+ - name: Primary earner under 65 in decile 5 with base elasticity
2
2
  period: 2023
3
3
  input:
4
+ age: 50
4
5
  employment_income_before_lsr: 50_000
5
6
  self_employment_income_before_lsr: 0
6
7
  gov.simulation.labor_supply_responses.elasticities.substitution.all: 0
8
+ gov.simulation.labor_supply_responses.elasticities.substitution.by_position_and_decile.primary.5: 0.15
9
+ gov.simulation.labor_supply_responses.elasticities.substitution.age_multiplier_over_threshold: 2.0
7
10
  output:
8
- substitution_elasticity: 0
11
+ substitution_elasticity: 0.15
9
12
 
10
- - name: Global substitution elasticity override
13
+ - name: Primary earner 65+ in decile 5 with multiplier effect
11
14
  period: 2023
12
15
  input:
16
+ age: 70
13
17
  employment_income_before_lsr: 50_000
14
18
  self_employment_income_before_lsr: 0
15
- gov.simulation.labor_supply_responses.elasticities.substitution.all: 0.25
16
- output:
17
- substitution_elasticity: 0.25
18
-
19
- - name: Primary earner in decile 1 with default parameters
20
- period: 2023
21
- input:
22
- employment_income_before_lsr: 10_000
23
- self_employment_income_before_lsr: 0
24
19
  gov.simulation.labor_supply_responses.elasticities.substitution.all: 0
25
- gov.simulation.labor_supply_responses.elasticities.substitution.by_position_and_decile.primary.1: 0.1
26
- gov.simulation.labor_supply_responses.elasticities.substitution.by_position_and_decile.secondary: 0.2
20
+ gov.simulation.labor_supply_responses.elasticities.substitution.by_position_and_decile.primary.5: 0.15
21
+ gov.simulation.labor_supply_responses.elasticities.substitution.age_multiplier_over_threshold: 2.0
27
22
  output:
28
- substitution_elasticity: 0 # TODO: Debug why single person isn't getting primary earner elasticity
29
-
30
- - name: Negative total earnings should have zero elasticity
31
- period: 2023
32
- input:
33
- employment_income_before_lsr: 50_000
34
- self_employment_income_before_lsr: -60_000 # Net negative earnings
35
- gov.simulation.labor_supply_responses.elasticities.substitution.all: 0
36
- gov.simulation.labor_supply_responses.elasticities.substitution.by_position_and_decile.primary.1: 0.2
37
- output:
38
- substitution_elasticity: 0 # max_(earnings, 0) = 0, so elasticity = 0
39
-
40
- - name: Positive net earnings after self-employment loss
41
- period: 2023
42
- input:
43
- employment_income_before_lsr: 50_000
44
- self_employment_income_before_lsr: -5_000 # Net positive earnings = 45_000
45
- gov.simulation.labor_supply_responses.elasticities.substitution.all: 0
46
- gov.simulation.labor_supply_responses.elasticities.substitution.by_position_and_decile.primary.5: 0.15 # 45k falls in decile 5
47
- output:
48
- substitution_elasticity: 0.15
23
+ substitution_elasticity: 0.30
@@ -7,4 +7,30 @@ class income_elasticity(Variable):
7
7
  label = "income elasticity of labor supply"
8
8
  unit = "/1"
9
9
  definition_period = YEAR
10
- adds = ["gov.simulation.labor_supply_responses.elasticities.income"]
10
+ reference = [
11
+ "https://www.cbo.gov/publication/43675",
12
+ "https://www.cbo.gov/publication/43676",
13
+ "https://academic.oup.com/restud/article-abstract/72/2/395/1558553",
14
+ ]
15
+
16
+ def formula(person, period, parameters):
17
+ p = parameters(
18
+ period
19
+ ).gov.simulation.labor_supply_responses.elasticities.income
20
+
21
+ # Check if global override is set
22
+ if p.all != 0:
23
+ return p.all
24
+
25
+ # Get base income elasticity
26
+ base_elasticity = p.base
27
+
28
+ # Apply age multiplier for individuals at or above age threshold
29
+ age = person("age", period.this_year)
30
+ age_multiplier = where(
31
+ age >= p.age_threshold,
32
+ p.age_multiplier_over_threshold,
33
+ 1.0, # No multiplier for under threshold
34
+ )
35
+
36
+ return base_elasticity * age_multiplier
@@ -7,12 +7,18 @@ class substitution_elasticity(Variable):
7
7
  label = "substitution elasticity of labor supply"
8
8
  unit = "/1"
9
9
  definition_period = YEAR
10
+ reference = [
11
+ "https://www.cbo.gov/publication/43675",
12
+ "https://www.cbo.gov/publication/43676",
13
+ "https://academic.oup.com/restud/article-abstract/72/2/395/1558553",
14
+ ]
10
15
 
11
16
  def formula(person, period, parameters):
12
17
  p = parameters(
13
18
  period
14
19
  ).gov.simulation.labor_supply_responses.elasticities.substitution
15
20
 
21
+ # Check for global override
16
22
  if p.all != 0:
17
23
  return p.all
18
24
 
@@ -30,6 +36,7 @@ class substitution_elasticity(Variable):
30
36
  1_726e3,
31
37
  ]
32
38
 
39
+ # Calculate earnings
33
40
  raw_earnings = add(
34
41
  person,
35
42
  period,
@@ -38,41 +45,59 @@ class substitution_elasticity(Variable):
38
45
  "self_employment_income_before_lsr",
39
46
  ],
40
47
  )
48
+ # Use max_ to prevent negative earnings from causing issues
41
49
  earnings = max_(raw_earnings, 0)
42
- earnings_decile = (
43
- np.searchsorted(EARNINGS_DECILE_MARKERS, earnings) + 1
50
+ # Use searchsorted with side='right' to bin earnings into deciles
51
+ # This gives decile numbers 1-10 directly without needing to add 1
52
+ earnings_decile = np.searchsorted(
53
+ EARNINGS_DECILE_MARKERS, earnings, side="right"
44
54
  )
45
55
 
56
+ # Determine if primary earner (highest earner in tax unit)
46
57
  tax_unit = person.tax_unit
47
- # Primary earner == highest earner in tax unit
48
58
  max_earnings_in_unit = tax_unit.max(earnings)
49
59
  is_primary_earner = earnings == max_earnings_in_unit
50
60
 
51
- elasticities = np.zeros_like(earnings)
61
+ # Get base elasticity from position and decile
62
+ base_elasticity = np.zeros_like(earnings, dtype=float)
52
63
 
53
64
  # Handle zero earnings first
54
65
  zero_earnings = earnings == 0
55
- elasticities[zero_earnings] = 0
66
+ base_elasticity[zero_earnings] = 0
56
67
 
57
- # For non-zero earnings, assign elasticities
68
+ # For non-zero earnings, assign base elasticities
58
69
  non_zero_earnings = earnings > 0
59
70
 
60
71
  if np.any(non_zero_earnings):
61
- # First assign primary earner elasticities by decile
62
- decile_elasticities = [
63
- p.by_position_and_decile.primary._children[str(i + 1)]
64
- for i in range(10)
65
- ]
66
- for i in range(10):
72
+ # Primary earners by decile
73
+ for i in range(10): # Iterate through deciles 1-10
74
+ decile_num = i + 1 # Decile numbers are 1-10
67
75
  mask = (
68
76
  non_zero_earnings
69
- & (earnings_decile == i + 1)
77
+ & (earnings_decile == decile_num)
70
78
  & is_primary_earner
71
79
  )
72
- elasticities[mask] = decile_elasticities[i]
80
+ if np.any(mask):
81
+ # Access parameter using getattr to handle both normal and test scenarios
82
+ decile_param = getattr(
83
+ p.by_position_and_decile.primary, str(decile_num)
84
+ )
85
+ # Handle both Parameter objects and raw values (when overridden in tests)
86
+ param_value = (
87
+ decile_param(period)
88
+ if callable(decile_param)
89
+ else decile_param
90
+ )
91
+ base_elasticity[mask] = param_value
73
92
 
74
- # Then assign secondary earner elasticity where applicable
75
- secondary_mask = non_zero_earnings & ~is_primary_earner
76
- elasticities[secondary_mask] = p.by_position_and_decile.secondary
93
+ # Secondary earners get zero elasticity (only primary earners have non-zero values)
77
94
 
78
- return elasticities
95
+ # Apply age multiplier for individuals at or above age threshold
96
+ age = person("age", period.this_year)
97
+ age_multiplier = where(
98
+ age >= p.age_threshold,
99
+ p.age_multiplier_over_threshold,
100
+ 1.0, # No multiplier for under threshold
101
+ )
102
+
103
+ return base_elasticity * age_multiplier
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: policyengine-us
3
- Version: 1.425.7
3
+ Version: 1.426.0
4
4
  Summary: Add your description here.
5
5
  Author-email: PolicyEngine <hello@policyengine.org>
6
6
  License-File: LICENSE
@@ -918,9 +918,15 @@ policyengine_us/parameters/gov/simulation/capital_gains_responses/elasticity.yam
918
918
  policyengine_us/parameters/gov/simulation/labor_supply_responses/bounds/README.md,sha256=o3Ma9FY6zLzF3qYli3J52cv2Dw1WGvJUp9kDuXkDx-U,9
919
919
  policyengine_us/parameters/gov/simulation/labor_supply_responses/bounds/effective_wage_rate_change.yaml,sha256=MtDdf5nH1N3-snStnkvh9JsfWlLfP0aKiTsBz23Yi7I,179
920
920
  policyengine_us/parameters/gov/simulation/labor_supply_responses/bounds/income_change.yaml,sha256=G3jBCDCS9aVPm_fIk7MroG3eeHC9R4FHFvyvFylhzXI,157
921
- policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/README.md,sha256=jlWjpmYXUItsM3twGtNyUkbKuAaE8O9GjWcIF0bVAlg,14
922
- policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/income.yaml,sha256=8368gwpFH5jRPP3JOsZTmPrOoqN1d_WWiEwuKxDTrJQ,209
923
- policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/substitution.yaml,sha256=q3b4i70HTPpRPbKU95QCRxL0wVxzi9_PRqDhGznd7Cs,1869
921
+ policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/README.md,sha256=p_ZppvSIgEdLOXZGN3PagygmqYfKlwaZDUWKakceNpc,7843
922
+ policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/income/age_multiplier_over_threshold.yaml,sha256=QpIwprtWdpjLbGp0PkSWdCJPEmPs5l-Kc7MZP3rY3Lc,769
923
+ policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/income/age_threshold.yaml,sha256=F1-N4w7C3L_NksKdcxrYbHtD4Qc9GF9IVNJ8mxbCeJQ,644
924
+ policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/income/all.yaml,sha256=yfIGr7j1GjiQgEfgvVW1BINWMhyDLQMzSO21Eq96imE,293
925
+ policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/income/base.yaml,sha256=IDdYy_GyuF-6vqZ2vtMHoJfA9X0GCNuOCxVQ8N8-D24,648
926
+ policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/substitution/age_multiplier_over_threshold.yaml,sha256=t16tyufELBFdV-UItR7BXf8BVgXVoXkdnxugJbRzaO4,802
927
+ policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/substitution/age_threshold.yaml,sha256=F1-N4w7C3L_NksKdcxrYbHtD4Qc9GF9IVNJ8mxbCeJQ,644
928
+ policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/substitution/all.yaml,sha256=JjAYkRUs6zcy8bPrP6HSiKqpCTxXhDFHKPrTLBW14NY,309
929
+ policyengine_us/parameters/gov/simulation/labor_supply_responses/elasticities/substitution/by_position_and_decile.yaml,sha256=e0qsuZygCDZ_EbgGydfEYOSRxwU-AOpZfoblxsXTVkQ,2070
924
930
  policyengine_us/parameters/gov/ssa/README.md,sha256=Rcxn6NYTRCJYhktw0-I4QxgcDsYYsqCKabBb7EJq4gw,39
925
931
  policyengine_us/parameters/gov/ssa/nawi.yaml,sha256=Dbb8NPv7AEmaKPhHgDsWA5D54Q2aJPkr0OSccgDR4bw,4197
926
932
  policyengine_us/parameters/gov/ssa/uprating.yaml,sha256=qR8giFlpvxYxaeOopHqy8BiGT1OQQEz2X09elmbLNl4,1447
@@ -3783,9 +3789,10 @@ policyengine_us/tests/policy/baseline/gov/local/tx/harris/in_harris_county_tx.ya
3783
3789
  policyengine_us/tests/policy/baseline/gov/local/tx/harris/rides/tx_harris_rides_eligible.yaml,sha256=eOBTzdhMtzwr3QAUURqqpNKgAsk7z9HXl543qD8oqUo,796
3784
3790
  policyengine_us/tests/policy/baseline/gov/local/tx/harris/rides/tx_harris_rides_subsidy.yaml,sha256=m4V1kGW3ID_znp8EWOLZBX_PqyWhs371OJHZCY1VZq0,852
3785
3791
  policyengine_us/tests/policy/baseline/gov/simulation/labor_supply_response/employment_income_behavioral_response.yaml,sha256=UyVqLHduKDRHuIE-wJS0czNGyEDmTNq2gIiVoW-kEpA,1695
3786
- policyengine_us/tests/policy/baseline/gov/simulation/labor_supply_response/income_elasticity.yaml,sha256=_NCgAPoomWP-Hp563Ih5mBtN7VTywrXmtx0BhFO5r0c,336
3787
- policyengine_us/tests/policy/baseline/gov/simulation/labor_supply_response/labor_supply_behavioral_response.yaml,sha256=FqBkZcYs-EFj-pNtUAtW85ZyGn_riZ3_pH-8DP90ORU,1551
3788
- policyengine_us/tests/policy/baseline/gov/simulation/labor_supply_response/substitution_elasticity.yaml,sha256=Cfb3-vmZxtLJdDIimJsByRDqdYh1ds5lBCQVgShDDtQ,2032
3792
+ policyengine_us/tests/policy/baseline/gov/simulation/labor_supply_response/income_elasticity.yaml,sha256=JZAa-i5I9xkMgd_SPu5ifYfDAKnzYmv_B8nkvXfnhXE,747
3793
+ policyengine_us/tests/policy/baseline/gov/simulation/labor_supply_response/integration.yaml,sha256=hnRnR_mlswcCK-n7GTStDAa5bBME3nKR_VDIlIUoIx8,2385
3794
+ policyengine_us/tests/policy/baseline/gov/simulation/labor_supply_response/labor_supply_behavioral_response.yaml,sha256=5Zqh9KxGEg9u1CgIrxjFD6G-qYzZ7Ut9B9tdkwIdv5Q,799
3795
+ policyengine_us/tests/policy/baseline/gov/simulation/labor_supply_response/substitution_elasticity.yaml,sha256=LriCrDWZbyNX4g6bz__N1qesWuBgtgB_4vDbiFobIQQ,1022
3789
3796
  policyengine_us/tests/policy/baseline/gov/ssa/ss/never_eligible_for_social_security_benefits.yaml,sha256=-l0F8L_YCNpNfwBtZLxEfPOCPea4ZHfGNAHRw3ozkYk,587
3790
3797
  policyengine_us/tests/policy/baseline/gov/ssa/ssi/integration_tests.yaml,sha256=CLLMuqslIKd8niom1AL0CrhnkGlTH30rF0MOfLTA_3o,8398
3791
3798
  policyengine_us/tests/policy/baseline/gov/ssa/ssi/is_ssi_aged_blind_disabled.yaml,sha256=tyim2uGywNbV57pL2M6DO2kH1I2lXZS6P2Vab2zyIRA,470
@@ -6205,13 +6212,13 @@ policyengine_us/variables/gov/local/tx/harris/rides/tx_harris_rides_eligible.py,
6205
6212
  policyengine_us/variables/gov/local/tx/harris/rides/tx_harris_rides_subsidy.py,sha256=fuMRW0_3Lh0q-9OJi3DyY0l5dJA_4ejj1fbIEl5xLWg,875
6206
6213
  policyengine_us/variables/gov/simulation/capital_gains_responses.py,sha256=Sjie6qu1LQIb5Ktk4iGXuEoHxQ0soSIenameuxHdyX0,5947
6207
6214
  policyengine_us/variables/gov/simulation/labor_supply_response/employment_income_behavioral_response.py,sha256=_gErh7LpoOhZ-_b67cg3qtNgexecofttE4fAw0-htMg,840
6208
- policyengine_us/variables/gov/simulation/labor_supply_response/income_elasticity.py,sha256=QIi7EX2EmvW3S2nYP_6vJ37uU9YEsM249QGhudBoQ7Q,286
6215
+ policyengine_us/variables/gov/simulation/labor_supply_response/income_elasticity.py,sha256=yVNHx5hJ0rI90J0OCyWeZ7xFWusK0Wotj-HIeZ1iPKE,1077
6209
6216
  policyengine_us/variables/gov/simulation/labor_supply_response/income_elasticity_lsr.py,sha256=GpHk3ZAYfsI6qXWD8iTJ_IRvaoYHg4s2gNeTl22kd2Y,732
6210
6217
  policyengine_us/variables/gov/simulation/labor_supply_response/labor_supply_behavioral_response.py,sha256=KpnZMqjMoS0tpDKT6TDp0ev9moxka5lisjqDxoHdOWA,2422
6211
6218
  policyengine_us/variables/gov/simulation/labor_supply_response/relative_income_change.py,sha256=EOCmle2VTADu0A0VXmSZk38SzVeH7oKPe-4Noyvl62c,1346
6212
6219
  policyengine_us/variables/gov/simulation/labor_supply_response/relative_wage_change.py,sha256=Lhn-zdGyIevS04jF3roS4101GF70N1-FPpNeKimW2iU,1361
6213
6220
  policyengine_us/variables/gov/simulation/labor_supply_response/self_employment_income_behavioral_response.py,sha256=zdJjfUOVbVRQEtcHVIssvxkS5D3BZlnn6GxI9AIA_mQ,352
6214
- policyengine_us/variables/gov/simulation/labor_supply_response/substitution_elasticity.py,sha256=2_Wo5XPqMiC4LnSfbcSVj-P2QWUCCcGBd8r95c0Afjw,2282
6221
+ policyengine_us/variables/gov/simulation/labor_supply_response/substitution_elasticity.py,sha256=eet1LtMM0h0E1cwLtNifajVA7NdKIz-LIr7YKs7GFHs,3528
6215
6222
  policyengine_us/variables/gov/simulation/labor_supply_response/substitution_elasticity_lsr.py,sha256=TH5bDW97u1N5pIr82eVve5DDt7A10kg5zAipjodFbd0,768
6216
6223
  policyengine_us/variables/gov/ssa/ss/never_eligible_for_social_security_benefits.py,sha256=Yc8a0ySF-Ddijb5CMJBNklqUSW4VQH7d5mNSJClo_6c,547
6217
6224
  policyengine_us/variables/gov/ssa/ss/social_security.py,sha256=J5zzCleYk_K5eMCtrlvzAlOXyLwJESqzcnF7w4ubiKU,438
@@ -8842,8 +8849,8 @@ policyengine_us/variables/input/farm_income.py,sha256=BEKxYmHNNnWJAAvULl5qZJigy5
8842
8849
  policyengine_us/variables/input/geography.py,sha256=Ux0ueAf0rhZaflyEqz81UuXP3xKCKBDvoO3CrKhiQEc,5421
8843
8850
  policyengine_us/variables/input/self_employment_income.py,sha256=PwsGz8R4lRikKWUYOhsC0qosNNLXq4f5SQmfw4S3mk8,511
8844
8851
  policyengine_us/variables/input/self_employment_income_before_lsr.py,sha256=E8fcX9Nlyqz8dziHhQv_euutdmoIwFMMWePUwbbwv_w,379
8845
- policyengine_us-1.425.7.dist-info/METADATA,sha256=p_6Q5nKK9ePnzkpu12ajcKSPNfdOfYUl3bVpZS7f7rQ,1649
8846
- policyengine_us-1.425.7.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
8847
- policyengine_us-1.425.7.dist-info/entry_points.txt,sha256=MLaqNyNTbReALyKNkde85VkuFFpdPWAcy8VRG1mjczc,57
8848
- policyengine_us-1.425.7.dist-info/licenses/LICENSE,sha256=2N5ReRelkdqkR9a-KP-y-shmcD5P62XoYiG-miLTAzo,34519
8849
- policyengine_us-1.425.7.dist-info/RECORD,,
8852
+ policyengine_us-1.426.0.dist-info/METADATA,sha256=LhSAqmdjP1G85s5SDW_9soTRZWl4bmaNSk0NBeTFS7k,1649
8853
+ policyengine_us-1.426.0.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
8854
+ policyengine_us-1.426.0.dist-info/entry_points.txt,sha256=MLaqNyNTbReALyKNkde85VkuFFpdPWAcy8VRG1mjczc,57
8855
+ policyengine_us-1.426.0.dist-info/licenses/LICENSE,sha256=2N5ReRelkdqkR9a-KP-y-shmcD5P62XoYiG-miLTAzo,34519
8856
+ policyengine_us-1.426.0.dist-info/RECORD,,
@@ -1,70 +0,0 @@
1
- all:
2
- description: Percent change (of the change in the effective marginal wage) in labor supply given a 1% change in the effective marginal wage. This parameter overrides all other substitution elasticities if provided.
3
- values:
4
- 2020-01-01: 0
5
- metadata:
6
- unit: /1
7
- label: substitution elasticity of labor supply
8
- by_position_and_decile:
9
- metadata:
10
- label: by position and decile
11
- unit: /1
12
- propagate_metadata_to_children: true
13
- primary:
14
- metadata:
15
- label: primary adult
16
- 1:
17
- metadata:
18
- label: primary adult, 1st decile substitution elasticity
19
- values:
20
- 2020-01-01: 0
21
- 2:
22
- metadata:
23
- label: primary adult, 2nd decile substitution elasticity
24
- values:
25
- 2020-01-01: 0
26
- 3:
27
- metadata:
28
- label: primary adult, 3rd decile substitution elasticity
29
- values:
30
- 2020-01-01: 0
31
- 4:
32
- metadata:
33
- label: primary adult, 4th decile substitution elasticity
34
- values:
35
- 2020-01-01: 0
36
- 5:
37
- metadata:
38
- label: primary adult, 5th decile substitution elasticity
39
- values:
40
- 2020-01-01: 0
41
- 6:
42
- metadata:
43
- label: primary adult, 6th decile substitution elasticity
44
- values:
45
- 2020-01-01: 0
46
- 7:
47
- metadata:
48
- label: primary adult, 7th decile substitution elasticity
49
- values:
50
- 2020-01-01: 0
51
- 8:
52
- metadata:
53
- label: primary adult, 8th decile substitution elasticity
54
- values:
55
- 2020-01-01: 0
56
- 9:
57
- metadata:
58
- label: primary adult, 9th decile substitution elasticity
59
- values:
60
- 2020-01-01: 0
61
- 10:
62
- metadata:
63
- label: primary adult, 10th decile substitution elasticity
64
- values:
65
- 2020-01-01: 0
66
- secondary:
67
- metadata:
68
- label: secondary adult, all deciles substitution elasticity
69
- values:
70
- 2020-01-01: 0